#Part 1 The purpose of this notebook is to determine the selective benefit of glucosinolate and flavonoid compounds through plant competition by creating selection gradients. Selection gradients will involve final body mass as the proxy for fitness, but this will be replaced with fitness once the measurement is in. Concentration will be on the x-axis. This analysis will account for family and greenhouse location.

#Part 2 The second purpose is to determine if glucosinolates and flavonoids influence suscpetibility to pathogens and if this susceptibility results in increased fitness

#Read in and prep data

library(ggplot2)
library(lme4)
library(tidyr)
library(cowplot)
library(grid)
library(gridExtra)
library(dplyr)
source("GGPlot_Themes.R")

#read in data
rm(list=ls())
dat<-read.csv("DataSynthesis.csv")

#Remove maple controls data from the data set.
dat<-dat %>% filter(treatment!="mcnt") %>% droplevels()

#Assign family column
prefamily<-gsub("*.\\|","",dat$Tag)
dat$Family<-gsub("\\-.*","",prefamily)

#remove those with fertilizer treatment, and extra genotypes that are only in the alone treatment. 
dat<-dat[!grepl("i",dat$Sample,fixed=T),]

#Initializing columns to avoid constant error messages. 
dat$WhiteFungLogis<-NA
dat$BlackFungLogis<-NA

#Generating data frames with summarized leaf data (dat2) and summarized genotype data (dat3)

#Summarizing: Taking the mean value of leaves. This tibble contains data at the level of the plant means. 
dat2<-dat %>% group_by(Tag) %>% summarize(ChlorA=mean(ChlorA),ChlorB=mean(ChlorB),gluc_Conc=mean(gluc_Conc),flav_Conc=mean(flav_Conc),Family=first(Family),treatment=first(treatment),gh_row=first(gh_row),gh_bench=first(gh_bench),GM_TotalLeaf_Area=first(GM_TotalLeaf_Area),comp_number=first(comp_number),ThripsDam=mean(ThripsDam),WhiteFungDam=mean(WhiteFungDam),BlackPathDam=mean(BlackPathDam),Fern=mean(Fern),gh_col=first(gh_col))
dat

#All of these genotypes died (15 Genotypes).(We are simply missing a final measurement for e|JBCHY1-1-50|Q|240) That is only 3% Mortality. 
dead<-dat2[is.na(dat2$GM_TotalLeaf_Area),]

#Removing those with dead competitors from the garlic mustard treatment. ("e|JBCHY1-1-50|Q|240") did not die, we are simply missing the final measurement for it.

dead_competitors<-dead %>% filter(treatment=="gm",Tag!="e|JBCHY1-1-50|Q|240") %>% select(comp_number)

#Removing those with dead competitors from the analysis. 
dat2<-dat2 %>% filter(!comp_number %in% dead_competitors$comp_number)
dat<-dat %>% filter(!comp_number %in% dead_competitors$comp_number)


##Summarizing: Taking the mean value of plants. This tibble contains data at the level of the family means within each treatment. 
dat3<-dat2 %>% drop_na(GM_TotalLeaf_Area) %>% group_by(Family,treatment)  %>% summarize_if(is.numeric,mean)

#Searching for a fitness trade-off between the alone and interspecific competition treatment.

source("GGPlot_Themes.R")

#Calculating family means within treatment. 
TradeOff<-dat3 %>% filter(treatment=="a") %>% drop_na(GM_TotalLeaf_Area) %>%  select(LeafSizeAlone=GM_TotalLeaf_Area,Family,gluc_ConcAlone=gluc_Conc) %>% right_join(dat3 %>% filter(treatment=="m") %>%  select(LeafSizeMaple=GM_TotalLeaf_Area,Family,gluc_ConcMaple=gluc_Conc),by="Family")

#Calculating standard error of each family
StdErr<-dat2 %>% select(Family,treatment,GM_TotalLeaf_Area) %>% group_by(Family,treatment) %>% drop_na(GM_TotalLeaf_Area) %>%  summarize(StdErr=sd(GM_TotalLeaf_Area)/sqrt(length(GM_TotalLeaf_Area)),size=length(GM_TotalLeaf_Area))

#Shifting the data to be in long form. 
StdErr2<-StdErr %>% filter(treatment=="a") %>% select(StdErrAlone=StdErr,Family) %>% right_join(StdErr %>% filter(treatment=="m") %>% select(StdErrMaple=StdErr,Family),by="Family")

TradeOff2<-StdErr2 %>% left_join(TradeOff)
Joining, by = "Family"
#tiff("Selection_Figures/TradeOff.tiff", units="in", width=10, height=6, res=300)

ggplot(TradeOff2,aes(y=LeafSizeAlone,x=LeafSizeMaple))+
  geom_point()+
  geom_linerange(aes(ymin = LeafSizeAlone - StdErrAlone, 
                    ymax = LeafSizeAlone + StdErrAlone))+
  geom_errorbarh(aes(xmin = LeafSizeMaple - StdErrMaple,
                    xmax = LeafSizeMaple + StdErrMaple))+
theme_simple()+
  xlab("Performance with Maple")+
  ylab("Performance Alone")

#dev.off()
summary(lm(LeafSizeAlone~LeafSizeMaple,data=TradeOff2))

Call:
lm(formula = LeafSizeAlone ~ LeafSizeMaple, data = TradeOff2)

Residuals:
    Min      1Q  Median      3Q     Max 
-2019.5  -657.0  -265.5   907.7  2364.0 

Coefficients:
                Estimate Std. Error t value Pr(>|t|)    
(Intercept)   9590.03571 1616.26626   5.933 6.87e-06 ***
LeafSizeMaple   -0.00936    0.18258  -0.051     0.96    
---
Signif. codes:  0 ‘***’ 0.001 ‘**’ 0.01 ‘*’ 0.05 ‘.’ 0.1 ‘ ’ 1

Residual standard error: 1244 on 21 degrees of freedom
Multiple R-squared:  0.0001251, Adjusted R-squared:  -0.04749 
F-statistic: 0.002628 on 1 and 21 DF,  p-value: 0.9596

#Visualizing genetic variation and greenhouse variation, which will be controlled for.

#GH Bench
ggplot(dat2)+
  geom_point(aes(y=gluc_Conc,x=gh_bench,colour=as.factor(gh_bench)))


#GH Col
ggplot(dat2)+
  geom_point(aes(y=gluc_Conc,x=gh_col,colour=as.factor(gh_bench)))


#Investigating genetic differences by treatment
#gluc_Conc
boxplot(gluc_Conc~Family,data=dat2[dat2$treatment=="a",])

boxplot(gluc_Conc~Family,data=dat2[dat2$treatment=="m",])

boxplot(gluc_Conc~Family,data=dat2[dat2$treatment=="gm",])

#bodymass
boxplot(GM_TotalLeaf_Area~Family,data=dat2[dat2$treatment=="a",])

boxplot(GM_TotalLeaf_Area~Family,data=dat2[dat2$treatment=="m",])

boxplot(GM_TotalLeaf_Area~Family,data=dat2[dat2$treatment=="gm",])

#Modelling: What influences performance (total leaf area)

#With limited dataset containing flavonoid data
summary(lmer(GM_TotalLeaf_Area ~ treatment*gluc_Conc+Fern+BlackPathDam +flav_Conc + (1 | gh_bench/gh_col) ,data=dat2))
Linear mixed model fit by REML. t-tests use Satterthwaite's method [
lmerModLmerTest]
Formula: GM_TotalLeaf_Area ~ treatment * gluc_Conc + Fern + BlackPathDam +  
    flav_Conc + (1 | gh_bench/gh_col)
   Data: dat2

REML criterion at convergence: 6388.9

Scaled residuals: 
    Min      1Q  Median      3Q     Max 
-3.2860 -0.5799  0.0639  0.5879  2.8024 

Random effects:
 Groups          Name        Variance Std.Dev.
 gh_col:gh_bench (Intercept)  588144   766.9  
 gh_bench        (Intercept) 1927899  1388.5  
 Residual                    7116930  2667.8  
Number of obs: 349, groups:  gh_col:gh_bench, 28; gh_bench, 5

Fixed effects:
                      Estimate Std. Error       df t value Pr(>|t|)    
(Intercept)           11705.91    2415.48   231.10   4.846 2.31e-06 ***
treatmentgm           -8139.74    2860.71   322.70  -2.845 0.004720 ** 
treatmentm            -8725.71    3156.29   328.21  -2.765 0.006022 ** 
gluc_Conc             -4183.45    2377.16   322.41  -1.760 0.079381 .  
Fern                   -122.68      56.51   330.47  -2.171 0.030631 *  
BlackPathDam           -162.81      41.42   328.48  -3.931 0.000103 ***
flav_Conc              2524.64    1213.99   338.49   2.080 0.038312 *  
treatmentgm:gluc_Conc  4230.28    2889.49   321.40   1.464 0.144165    
treatmentm:gluc_Conc   7832.59    3279.65   328.95   2.388 0.017493 *  
---
Signif. codes:  0 ‘***’ 0.001 ‘**’ 0.01 ‘*’ 0.05 ‘.’ 0.1 ‘ ’ 1

Correlation of Fixed Effects:
            (Intr) trtmntg trtmntm glc_Cn Fern   BlckPD flv_Cn trtmntg:_C
treatmentgm -0.767                                                       
treatmentm  -0.697  0.588                                                
gluc_Conc   -0.856  0.736   0.675                                        
Fern        -0.001 -0.009  -0.055  -0.032                                
BlackPathDm -0.019  0.093   0.046  -0.053 -0.078                         
flav_Conc   -0.179  0.084   0.071  -0.269  0.049  0.102                  
trtmntgm:_C  0.751 -0.991  -0.577  -0.737  0.002 -0.094 -0.063           
trtmntm:g_C  0.669 -0.564  -0.993  -0.655  0.047 -0.042 -0.069  0.559    

#Model Diagnostics.

#Visualization – The conditional benefit of glucosinolates (allelopathy)


aloneSlope<-function(x){
  y=-3966.60*x+ 13751.96
  return(y)
}
mapleSlope<-function(x){
  y=(-4254.2+8417.95)*x+13751.96 -9243.71
  return(y)
}

mustardSlope<-function(x){
  y=+13751.96-8376.67#Non significant slope
  return(y)
}

minM<-min(dat2$gluc_Conc[dat2$treatment=="m"],na.rm = T)
maxM<-max(dat2$gluc_Conc[dat2$treatment=="m"],na.rm = T)

minA<-min(dat2$gluc_Conc[dat2$treatment=="a"],na.rm = T)
maxA<-max(dat2$gluc_Conc[dat2$treatment=="a"],na.rm = T)

minG<-min(dat2$gluc_Conc[dat2$treatment=="gm"],na.rm = T)
maxG<-max(dat2$gluc_Conc[dat2$treatment=="gm"],na.rm = T)

#tiff("Selection_Figures/Gluc_Benefit.tiff", units="in", width=10, height=6, res=300)
library(ggplot2)
ggplot(dat2)+
  geom_point(aes(y=GM_TotalLeaf_Area,x=gluc_Conc,colour=treatment),size=2)+
  geom_segment(x=minA,xend=maxA,y=aloneSlope(minA),yend=aloneSlope(maxA),colour="#009E73",size=1.5)+
    geom_segment(x=minM,xend=maxM,y=mapleSlope(minM),yend=mapleSlope(maxM),colour="#E69F00",size=1.5)+
  geom_segment(x=minG,xend=maxG,y=mustardSlope(minG),yend=mustardSlope(maxG),colour="#56B4E9",size=1.5)+theme_simple()+
  ylab(bquote(bold("Performance\n(Total Leaf Area "~(mm^2)~")")))+xlab(bquote(bold("[Total Glucosinolate] " (mg/ml))))+
  scale_colour_manual(values=c("#009E73","#56B4E9","#E69F00"),labels=c("Alone","Garlic Mustard","Maple"))

#dev.off()

minA
[1] 0.7713421

#Visualizing the effect of flavonoid on fitness.

summary(lmer(GM_TotalLeaf_Area~treatment+flav_Conc+BlackPathDam+Fern+(1|Family)+(1|gh_bench/gh_col), data=dat2))
Linear mixed model fit by REML. t-tests use Satterthwaite's method [
lmerModLmerTest]
Formula: GM_TotalLeaf_Area ~ treatment + flav_Conc + BlackPathDam + Fern +  
    (1 | Family) + (1 | gh_bench/gh_col)
   Data: dat2

REML criterion at convergence: 6473.8

Scaled residuals: 
    Min      1Q  Median      3Q     Max 
-3.1990 -0.6278  0.0560  0.6004  2.6495 

Random effects:
 Groups          Name        Variance Std.Dev.
 gh_col:gh_bench (Intercept)  660632   812.8  
 Family          (Intercept)  210327   458.6  
 gh_bench        (Intercept) 1767108  1329.3  
 Residual                    7165525  2676.8  
Number of obs: 350, groups:  gh_col:gh_bench, 28; Family, 23; gh_bench, 5

Fixed effects:
             Estimate Std. Error       df t value Pr(>|t|)    
(Intercept)   7682.44    1156.80    33.30   6.641 1.42e-07 ***
treatmentgm  -3897.18     385.80   333.76 -10.102  < 2e-16 ***
treatmentm   -1133.35     364.02   330.79  -3.113  0.00201 ** 
flav_Conc     2471.19    1040.31   341.75   2.375  0.01808 *  
BlackPathDam   -77.69      30.19   335.79  -2.573  0.01050 *  
Fern          -125.75      57.56   330.33  -2.185  0.02962 *  
---
Signif. codes:  0 ‘***’ 0.001 ‘**’ 0.01 ‘*’ 0.05 ‘.’ 0.1 ‘ ’ 1

Correlation of Fixed Effects:
            (Intr) trtmntg trtmntm flv_Cn BlckPD
treatmentgm -0.294                              
treatmentm  -0.260  0.475                       
flav_Conc   -0.795  0.198   0.137               
BlackPathDm -0.163 -0.043   0.021   0.090       
Fern        -0.035 -0.037  -0.073   0.030 -0.156
aloneSlope<-function(x){
  y=2024.62 *x+  8035.30
  return(y)
}
mapleSlope<-function(x){
  y=(2024.62 )*x+ 8035.30 -1067.02
  return(y)
}

mustardSlope<-function(x){
  y=2024.62 *x+ 8035.30-3746.91#Non significant slope
  return(y)
}

minM<-min(dat2$flav_Conc[dat2$treatment=="m"],na.rm = T)
maxM<-max(dat2$flav_Conc[dat2$treatment=="m"],na.rm = T)

minA<-min(dat2$flav_Conc[dat2$treatment=="a"],na.rm = T)
maxA<-max(dat2$flav_Conc[dat2$treatment=="a"],na.rm = T)

minG<-min(dat2$flav_Conc[dat2$treatment=="gm"],na.rm = T)
maxG<-max(dat2$flav_Conc[dat2$treatment=="gm"],na.rm = T)


#tiff("Selection_Figures/Flav_Benefit.tiff", units="in", width=10, height=6, res=300)
ggplot(dat2)+
  geom_point(aes(y=GM_TotalLeaf_Area,x=flav_Conc,colour=treatment),size=2)+
  geom_segment(x=minA,xend=maxA,y=aloneSlope(minA),yend=aloneSlope(maxA),colour="#009E73",size=1.5)+
    geom_segment(x=minM,xend=maxM,y=mapleSlope(minM),yend=mapleSlope(maxM),colour="#E69F00",size=1.5)+theme_simple()+
  geom_segment(x=minG,xend=maxG,y=mustardSlope(minG),yend=mustardSlope(maxG),colour="#56B4E9",size=1.5)+theme_simple()+
  ylab(bquote(bold("Performance\n(Total Leaf Area "~(mm^2)~")")))+xlab(bquote(bold("[Total Flavonoid] " (mg/ml))))+
  scale_colour_manual(values=c("#009E73","#56B4E9","#E69F00"),labels=c("Alone","Garlic Mustard","Maple"))

#dev.off()

#Visualization– The Detriment of pathogens and ferns

summary(lmer(GM_TotalLeaf_Area ~ Fern+
(1 | Family) + (1 | gh_bench/gh_col) ,data=dat2))
boundary (singular) fit: see ?isSingular
Linear mixed model fit by REML. t-tests use Satterthwaite's method [
lmerModLmerTest]
Formula: GM_TotalLeaf_Area ~ Fern + (1 | Family) + (1 | gh_bench/gh_col)
   Data: dat2

REML criterion at convergence: 9607.6

Scaled residuals: 
     Min       1Q   Median       3Q      Max 
-2.85397 -0.70508  0.04291  0.74619  3.11787 

Random effects:
 Groups          Name        Variance Std.Dev.
 gh_col:gh_bench (Intercept)    96624  310.8  
 Family          (Intercept)        0    0.0  
 gh_bench        (Intercept)  2947085 1716.7  
 Residual                    10094168 3177.1  
Number of obs: 507, groups:  gh_col:gh_bench, 28; Family, 23; gh_bench, 5

Fixed effects:
            Estimate Std. Error       df t value Pr(>|t|)    
(Intercept) 7896.911    807.136    4.071   9.784 0.000561 ***
Fern        -226.020     56.097  432.082  -4.029 6.61e-05 ***
---
Signif. codes:  0 ‘***’ 0.001 ‘**’ 0.01 ‘*’ 0.05 ‘.’ 0.1 ‘ ’ 1

Correlation of Fixed Effects:
     (Intr)
Fern -0.055
convergence code: 0
boundary (singular) fit: see ?isSingular
a<-ggplot(dat2)+
  geom_point(aes(y=GM_TotalLeaf_Area,x=BlackPathDam))+theme_simple_multiCol()+
  geom_abline(intercept=8281.76,slope = -105.28,size=1.5)+
  xlab("Black Pathogen Damage")

b<-ggplot(dat2[dat2$WhiteFungDam<30,])+
  geom_point(aes(y=GM_TotalLeaf_Area,x=WhiteFungDam),colour="#999999")+theme_simple_multiCol()+
  theme(axis.title.x = element_text(color = "#999999", size = 16, face = "bold",margin=margin(3,0,3,0)),
      )+xlab("Powdery Mildew Damage")

c<-ggplot(dat2)+
  geom_point(aes(y=GM_TotalLeaf_Area,x=ThripsDam),colour="#E69F00")+theme_simple_multiCol()+xlab("Thrips Damage")+
    theme(axis.title.x = element_text(color = "#E69F00", size = 16, face = "bold",margin=margin(3,0,3,0)))

d<-ggplot(dat2)+
  geom_point(aes(y=GM_TotalLeaf_Area,x=Fern),colour="#009E73")+theme_simple_multiCol()+xlab("Fern Abundance")+
   geom_abline(intercept=8003.44,slope = -179.47,size=1.5,color="#009E73")+
  theme(axis.title.x = element_text(color = "#009E73", size = 16, face = "bold",margin=margin(3,0,3,0)))
   


plot<-plot_grid(a, b,ncol=2,rel_widths = c(1,1))
Removed 61 rows containing missing values (geom_point).Removed 61 rows containing missing values (geom_point).
plot2<-plot_grid(d, c,ncol=2,rel_widths = c(1,1))
Removed 9 rows containing missing values (geom_point).Removed 61 rows containing missing values (geom_point).
plot3<-plot_grid(a,b,c,d,ncol=2,rel_widths = c(1,1))
Removed 61 rows containing missing values (geom_point).Removed 61 rows containing missing values (geom_point).Removed 61 rows containing missing values (geom_point).Removed 9 rows containing missing values (geom_point).
plot4<-plot_grid(a,b,c,ncol=1,rel_widths = c(1,1,1))
Removed 61 rows containing missing values (geom_point).Removed 61 rows containing missing values (geom_point).Removed 61 rows containing missing values (geom_point).
plot5<-plot_grid(a,b,c,ncol=3,rel_widths = c(1,1,1))
Removed 61 rows containing missing values (geom_point).Removed 61 rows containing missing values (geom_point).Removed 61 rows containing missing values (geom_point).
y.grob<-textGrob(bquote(bold("Shoot Area "(mm^2))),gp=gpar(fontface="bold",fontsize=20),rot=90)


#tiff("Selection_Figures/PathogenEffect.tiff", units="in", width=14, height=6, res=300)
grid.arrange(plot,left=y.grob)

#dev.off()

#tiff("Selection_Figures/PathogenEffect2.tiff", units="in", width=14, height=6, res=300)
grid.arrange(plot2,left=y.grob)

#dev.off()

#tiff("Selection_Figures/PathogenEffect3.tiff", units="in", width=14, height=10, res=300)
grid.arrange(plot3,left=y.grob)

#dev.off

grid.arrange(plot4,left=y.grob)



#tiff("Selection_Figures/PathogenEffect5.tiff", units="in", width=16, height=5, res=300)
grid.arrange(plot5,left=y.grob)

#dev.off

#tiff("Selection_Figures/FernEffect.tiff", units="in", width=8, height=6, res=300)
grid.arrange(d,left=y.grob)

#dev.off

#—————————————– #Part 2

#Influence of gluc and flav on defence.

hist(dat$Fern)
Warning messages:
1: Unknown or uninitialised column: `WhiteFungLogis`. 
2: Unknown or uninitialised column: `WhiteFungLogis`. 
3: Unknown or uninitialised column: `WhiteFungLogis`. 
4: Unknown or uninitialised column: `WhiteFungLogis`. 
5: Unknown or uninitialised column: `WhiteFungLogis`. 
6: Unknown or uninitialised column: `WhiteFungLogis`. 
7: Unknown or uninitialised column: `WhiteFungLogis`. 
8: Unknown or uninitialised column: `WhiteFungLogis`. 
9: Unknown or uninitialised column: `WhiteFungLogis`. 
10: Unknown or uninitialised column: `WhiteFungLogis`. 

#Modelling : Negative binomial on Thrips Damage.

summary(fit_g)$coef
$cond
             Estimate Std. Error   z value     Pr(>|z|)
(Intercept)  3.585466  0.5478174  6.545002 5.949477e-11
gluc_Conc   -2.793644  0.5756398 -4.853111 1.215394e-06

$zi
NULL

$disp
NULL

#Visualize: Thrips Damage predicted by glucosinolates and flavonoids


source("GGPlot_Themes.R")
Warning messages:
1: Unknown or uninitialised column: `WhiteFungLogis`. 
2: Unknown or uninitialised column: `WhiteFungLogis`. 
3: Unknown or uninitialised column: `WhiteFungLogis`. 
4: Unknown or uninitialised column: `WhiteFungLogis`. 
#Reversing link function, to estimate the data on the response scale. 
flavSlope=function(x){
  y=exp(-2.4621 *x+2.9746)
  return(y)
}
glucSlope=function(x){
  y=exp(-2.7869*x+3.5755)
  return(y)
}

#Determining x range to fit the line to 
flavplot=seq(min(dat$flav_Conc,na.rm = T),max(dat$flav_Conc,na.rm=T),length.out = 688)
glucplot=seq(min(dat$gluc_Conc,na.rm = T),max(dat$gluc_Conc,na.rm=T),length.out = 708)

#Calculating slope values
flavy<-flavSlope(flavplot)
glucy<-glucSlope(glucplot)


#tiff("Defence_Figures/FlavonoidThrips.tiff", units="in", width=10, height=6, res=300)
ggplot(dat[!is.na(dat$flav_Conc),])+
  geom_point(aes(y=ThripsDam,x=flav_Conc))+theme_simple()+
  geom_path(x=flavplot,y=flavy,size=1,colour="#999999")+
  
  scale_y_continuous(breaks=seq(0,70,5))+
  ylab("Thrips Damage")+
xlab(bquote(bold("[Total Flavonoid] " (mg/ml))))

#dev.off()

ggplot(dat[!is.na(dat$gluc_Conc),])+
  geom_point(aes(y=ThripsDam,x=gluc_Conc))+theme_simple()+
  geom_path(x=glucplot,y=glucy,size=1,colour="#999999")+
  
  scale_y_continuous(breaks=seq(0,70,5))+
  ylab("Thrips Damage")+
xlab(bquote(bold("[Total Glucosinolate] " (mg/ml))))

I think that a logistic regression is more appropriate for fungal abundance, because the count of fungal infection could be arbuitrary, especially when fungal patches were large or the leaf was completely covered.

#WhitePathDam – logistic regression

summary(fit.4) #Garlic mustard in the maple treatment have less occurence of fungal colonization. 
Generalized linear mixed model fit by maximum likelihood (Laplace
  Approximation) [glmerMod]
 Family: binomial  ( logit )
Formula: WhiteFungLogis ~ treatment + (1 | Family)
   Data: dat2

     AIC      BIC   logLik deviance df.resid 
   519.2    535.7   -255.6    511.2      451 

Scaled residuals: 
    Min      1Q  Median      3Q     Max 
-1.3885 -0.6239 -0.4587  0.9947  3.3329 

Random effects:
 Groups Name        Variance Std.Dev.
 Family (Intercept) 0.544    0.7376  
Number of obs: 455, groups:  Family, 23

Fixed effects:
            Estimate Std. Error z value Pr(>|z|)   
(Intercept)  -0.6835     0.2395  -2.854  0.00431 **
treatmentgm  -0.4101     0.2613  -1.569  0.11656   
treatmentm   -0.8037     0.2756  -2.916  0.00355 **
---
Signif. codes:  0 ‘***’ 0.001 ‘**’ 0.01 ‘*’ 0.05 ‘.’ 0.1 ‘ ’ 1

Correlation of Fixed Effects:
            (Intr) trtmntg
treatmentgm -0.509        
treatmentm  -0.474  0.447 

#Visualizing – effect of treatment on proportion of fungal abundance.

#Modelling: Negative Binomial – BlackPathDam

fit_2<-glmmTMB(BlackPathDam~treatment+flav_Conc+(1|Family/Tag),family=nbinom2,data=dat)
fit_2<-glmmTMB(BlackPathDam~treatment+flav_Conc+(1|Family/Tag),family=nbinom2,data=dat)
summary(fit_2)
 Family: nbinom2  ( log )
Formula:          BlackPathDam ~ treatment + flav_Conc + (1 | Family/Tag)
Data: dat

     AIC      BIC   logLik deviance df.resid 
  1822.7   1854.3   -904.3   1808.7      669 

Random effects:

Conditional model:
 Groups     Name        Variance Std.Dev.
 Tag:Family (Intercept) 1.4410   1.2004  
 Family     (Intercept) 0.2517   0.5017  
Number of obs: 676, groups:  Tag:Family, 423; Family, 23

Overdispersion parameter for nbinom2 family (): 0.641 

Conditional model:
            Estimate Std. Error z value Pr(>|z|)   
(Intercept)   0.7826     0.4876   1.605   0.1085   
treatmentgm   0.1049     0.2480   0.423   0.6724   
treatmentm   -0.5204     0.2500  -2.082   0.0374 * 
flav_Conc    -1.5060     0.5172  -2.912   0.0036 **
---
Signif. codes:  0 ‘***’ 0.001 ‘**’ 0.01 ‘*’ 0.05 ‘.’ 0.1 ‘ ’ 1

#Visualizing: The effect of treatment and flavonoid abundance on black pathogen abundance.

#Visualization – Black pathogen damage by treatment


plot2<-dat2 %>% drop_na(BlackPathDam) %>% group_by(treatment) %>% summarize(BlackPathAve=mean(BlackPathDam,na.rm=T))

ggplot(plot2)+
  geom_col(aes(x=treatment,y=BlackPathAve,fill=treatment))+theme_simple()+ylab("Black Pathogen Infection\n(spots/leaf)")+xlab("Treatment")+theme(legend.position = "none")

#Modelling: Negative Binomial- Fern abundance

summary(fit_1)$coef
$cond
              Estimate Std. Error   z value   Pr(>|z|)
(Intercept) -1.5549556  0.7070463 -2.199227 0.02786176
treatmentgm  1.1242568  0.5037953  2.231574 0.02564310
treatmentm   0.9924083  0.4284677  2.316180 0.02054844

$zi
NULL

$disp
NULL

#Visualizing — the distribution of pathogens by treatment.

#Visualizing – effect of flavonoids on fern abundance.


PoisSlope=function(x,int){
  y=exp((-0.02353 -3.12475)*x+int)
  return(y)
}
exp(-0.6571)

flavplot=seq(min(dat$flav_Conc,na.rm = T),max(dat$flav_Conc,na.rm=T),length.out = 707)

flavyA<-PoisSlope(flavplot,-2.1683)
flavyM<-PoisSlope(flavplot,-1.60546-2.81450)
flavyGM<-PoisSlope(flavplot,-2.1683-0.2786)


#tiff("Defence_Figures/FlavonoidFern.tiff", units="in", width=10, height=6, res=300)
ggplot(dat)+
  geom_point(aes(y=Fern,x=flav_Conc,colour=treatment))+theme_simple()+
 # geom_path(x=flavplot,y=flavyA,size=1,colour="#009E73")
  #geom_path(x=flavplot,y=flavyGM,size=1,colour="#56B4E9")
  geom_path(x=flavplot,y=flavyM,size=1,colour="#E69F00")+
      scale_colour_manual(values=c("#009E73","#56B4E9","#E69F00"),labels=c("Alone","Garlic Mustard","Maple"))+
  scale_y_continuous(breaks=c(0,5,10,15,20,25,30,35,40))+
  ylab("Fern Abundance")+
xlab(bquote(bold("[Total Flavonoid] " (mg/ml))))
#dev.off()

How can we know that healthy plants dont just exhibit more secondary compounds and not that those with more secondary compounds are healthier?

LS0tCnRpdGxlOiAiU2VsZWN0aXZlIGJlbmVmaXQgb2YgR2x1Y29zaW5vbGF0ZSBhbmQgZmxhdm9ub2lkcyIKb3V0cHV0OiBodG1sX25vdGVib29rCi0tLQojUGFydCAxClRoZSBwdXJwb3NlIG9mIHRoaXMgbm90ZWJvb2sgaXMgdG8gZGV0ZXJtaW5lIHRoZSBzZWxlY3RpdmUgYmVuZWZpdCBvZiBnbHVjb3Npbm9sYXRlIGFuZCBmbGF2b25vaWQgY29tcG91bmRzIHRocm91Z2ggcGxhbnQgY29tcGV0aXRpb24gYnkgY3JlYXRpbmcgc2VsZWN0aW9uIGdyYWRpZW50cy4gU2VsZWN0aW9uIGdyYWRpZW50cyB3aWxsIGludm9sdmUgZmluYWwgYm9keSBtYXNzIGFzIHRoZSBwcm94eSBmb3IgZml0bmVzcywgYnV0IHRoaXMgd2lsbCBiZSByZXBsYWNlZCB3aXRoIGZpdG5lc3Mgb25jZSB0aGUgbWVhc3VyZW1lbnQgaXMgaW4uIENvbmNlbnRyYXRpb24gd2lsbCBiZSBvbiB0aGUgeC1heGlzLiBUaGlzIGFuYWx5c2lzIHdpbGwgYWNjb3VudCBmb3IgZmFtaWx5IGFuZCBncmVlbmhvdXNlIGxvY2F0aW9uLiAKCiNQYXJ0IDIKVGhlIHNlY29uZCBwdXJwb3NlIGlzIHRvIGRldGVybWluZSBpZiBnbHVjb3Npbm9sYXRlcyBhbmQgZmxhdm9ub2lkcyBpbmZsdWVuY2Ugc3VzY3BldGliaWxpdHkgdG8gcGF0aG9nZW5zIGFuZCBpZiB0aGlzIHN1c2NlcHRpYmlsaXR5IHJlc3VsdHMgaW4gaW5jcmVhc2VkIGZpdG5lc3MKCgoKCiNSZWFkIGluIGFuZCBwcmVwIGRhdGEKYGBge3J9CmxpYnJhcnkoZ2dwbG90MikKbGlicmFyeShsbWU0KQpsaWJyYXJ5KHRpZHlyKQpsaWJyYXJ5KGNvd3Bsb3QpCmxpYnJhcnkoZ3JpZCkKbGlicmFyeShncmlkRXh0cmEpCmxpYnJhcnkoZHBseXIpCnNvdXJjZSgiR0dQbG90X1RoZW1lcy5SIikKCiNyZWFkIGluIGRhdGEKcm0obGlzdD1scygpKQpkYXQ8LXJlYWQuY3N2KCJEYXRhU3ludGhlc2lzLmNzdiIpCgojUmVtb3ZlIG1hcGxlIGNvbnRyb2xzIGRhdGEgZnJvbSB0aGUgZGF0YSBzZXQuCmRhdDwtZGF0ICU+JSBmaWx0ZXIodHJlYXRtZW50IT0ibWNudCIpICU+JSBkcm9wbGV2ZWxzKCkKCiNBc3NpZ24gZmFtaWx5IGNvbHVtbgpwcmVmYW1pbHk8LWdzdWIoIiouXFx8IiwiIixkYXQkVGFnKQpkYXQkRmFtaWx5PC1nc3ViKCJcXC0uKiIsIiIscHJlZmFtaWx5KQoKI3JlbW92ZSB0aG9zZSB3aXRoIGZlcnRpbGl6ZXIgdHJlYXRtZW50LCBhbmQgZXh0cmEgZ2Vub3R5cGVzIHRoYXQgYXJlIG9ubHkgaW4gdGhlIGFsb25lIHRyZWF0bWVudC4gCmRhdDwtZGF0WyFncmVwbCgiaSIsZGF0JFNhbXBsZSxmaXhlZD1UKSxdCgojSW5pdGlhbGl6aW5nIGNvbHVtbnMgdG8gYXZvaWQgY29uc3RhbnQgZXJyb3IgbWVzc2FnZXMuIApkYXQkV2hpdGVGdW5nTG9naXM8LU5BCmRhdCRCbGFja0Z1bmdMb2dpczwtTkEKYGBgCgoKI0dlbmVyYXRpbmcgZGF0YSBmcmFtZXMgd2l0aCBzdW1tYXJpemVkIGxlYWYgZGF0YSAoZGF0MikgYW5kIHN1bW1hcml6ZWQgZ2Vub3R5cGUgZGF0YSAoZGF0MykKYGBge3J9CiNTdW1tYXJpemluZzogVGFraW5nIHRoZSBtZWFuIHZhbHVlIG9mIGxlYXZlcy4gVGhpcyB0aWJibGUgY29udGFpbnMgZGF0YSBhdCB0aGUgbGV2ZWwgb2YgdGhlIHBsYW50IG1lYW5zLiAKZGF0MjwtZGF0ICU+JSBncm91cF9ieShUYWcpICU+JSBzdW1tYXJpemUoQ2hsb3JBPW1lYW4oQ2hsb3JBKSxDaGxvckI9bWVhbihDaGxvckIpLGdsdWNfQ29uYz1tZWFuKGdsdWNfQ29uYyksZmxhdl9Db25jPW1lYW4oZmxhdl9Db25jKSxGYW1pbHk9Zmlyc3QoRmFtaWx5KSx0cmVhdG1lbnQ9Zmlyc3QodHJlYXRtZW50KSxnaF9yb3c9Zmlyc3QoZ2hfcm93KSxnaF9iZW5jaD1maXJzdChnaF9iZW5jaCksR01fVG90YWxMZWFmX0FyZWE9Zmlyc3QoR01fVG90YWxMZWFmX0FyZWEpLGNvbXBfbnVtYmVyPWZpcnN0KGNvbXBfbnVtYmVyKSxUaHJpcHNEYW09bWVhbihUaHJpcHNEYW0pLFdoaXRlRnVuZ0RhbT1tZWFuKFdoaXRlRnVuZ0RhbSksQmxhY2tQYXRoRGFtPW1lYW4oQmxhY2tQYXRoRGFtKSxGZXJuPW1lYW4oRmVybiksZ2hfY29sPWZpcnN0KGdoX2NvbCkpCmRhdAoKI0FsbCBvZiB0aGVzZSBnZW5vdHlwZXMgZGllZCAoMTUgR2Vub3R5cGVzKS4oV2UgYXJlIHNpbXBseSBtaXNzaW5nIGEgZmluYWwgbWVhc3VyZW1lbnQgZm9yIGV8SkJDSFkxLTEtNTB8UXwyNDApIFRoYXQgaXMgb25seSAzJSBNb3J0YWxpdHkuIApkZWFkPC1kYXQyW2lzLm5hKGRhdDIkR01fVG90YWxMZWFmX0FyZWEpLF0KCiNSZW1vdmluZyB0aG9zZSB3aXRoIGRlYWQgY29tcGV0aXRvcnMgZnJvbSB0aGUgZ2FybGljIG11c3RhcmQgdHJlYXRtZW50LiAoImV8SkJDSFkxLTEtNTB8UXwyNDAiKSBkaWQgbm90IGRpZSwgd2UgYXJlIHNpbXBseSBtaXNzaW5nIHRoZSBmaW5hbCBtZWFzdXJlbWVudCBmb3IgaXQuCgpkZWFkX2NvbXBldGl0b3JzPC1kZWFkICU+JSBmaWx0ZXIodHJlYXRtZW50PT0iZ20iLFRhZyE9ImV8SkJDSFkxLTEtNTB8UXwyNDAiKSAlPiUgc2VsZWN0KGNvbXBfbnVtYmVyKQoKI1JlbW92aW5nIHRob3NlIHdpdGggZGVhZCBjb21wZXRpdG9ycyBmcm9tIHRoZSBhbmFseXNpcy4gCmRhdDI8LWRhdDIgJT4lIGZpbHRlcighY29tcF9udW1iZXIgJWluJSBkZWFkX2NvbXBldGl0b3JzJGNvbXBfbnVtYmVyKQpkYXQ8LWRhdCAlPiUgZmlsdGVyKCFjb21wX251bWJlciAlaW4lIGRlYWRfY29tcGV0aXRvcnMkY29tcF9udW1iZXIpCgoKIyNTdW1tYXJpemluZzogVGFraW5nIHRoZSBtZWFuIHZhbHVlIG9mIHBsYW50cy4gVGhpcyB0aWJibGUgY29udGFpbnMgZGF0YSBhdCB0aGUgbGV2ZWwgb2YgdGhlIGZhbWlseSBtZWFucyB3aXRoaW4gZWFjaCB0cmVhdG1lbnQuIApkYXQzPC1kYXQyICU+JSBkcm9wX25hKEdNX1RvdGFsTGVhZl9BcmVhKSAlPiUgZ3JvdXBfYnkoRmFtaWx5LHRyZWF0bWVudCkgICU+JSBzdW1tYXJpemVfaWYoaXMubnVtZXJpYyxtZWFuKQoKYGBgCgoKI1NlYXJjaGluZyBmb3IgYSBmaXRuZXNzIHRyYWRlLW9mZiBiZXR3ZWVuIHRoZSBhbG9uZSBhbmQgaW50ZXJzcGVjaWZpYyBjb21wZXRpdGlvbiB0cmVhdG1lbnQuIApgYGB7cn0Kc291cmNlKCJHR1Bsb3RfVGhlbWVzLlIiKQoKI0NhbGN1bGF0aW5nIGZhbWlseSBtZWFucyB3aXRoaW4gdHJlYXRtZW50LiAKVHJhZGVPZmY8LWRhdDMgJT4lIGZpbHRlcih0cmVhdG1lbnQ9PSJhIikgJT4lIGRyb3BfbmEoR01fVG90YWxMZWFmX0FyZWEpICU+JSAgc2VsZWN0KExlYWZTaXplQWxvbmU9R01fVG90YWxMZWFmX0FyZWEsRmFtaWx5LGdsdWNfQ29uY0Fsb25lPWdsdWNfQ29uYykgJT4lIHJpZ2h0X2pvaW4oZGF0MyAlPiUgZmlsdGVyKHRyZWF0bWVudD09Im0iKSAlPiUgIHNlbGVjdChMZWFmU2l6ZU1hcGxlPUdNX1RvdGFsTGVhZl9BcmVhLEZhbWlseSxnbHVjX0NvbmNNYXBsZT1nbHVjX0NvbmMpLGJ5PSJGYW1pbHkiKQoKI0NhbGN1bGF0aW5nIHN0YW5kYXJkIGVycm9yIG9mIGVhY2ggZmFtaWx5ClN0ZEVycjwtZGF0MiAlPiUgc2VsZWN0KEZhbWlseSx0cmVhdG1lbnQsR01fVG90YWxMZWFmX0FyZWEpICU+JSBncm91cF9ieShGYW1pbHksdHJlYXRtZW50KSAlPiUgZHJvcF9uYShHTV9Ub3RhbExlYWZfQXJlYSkgJT4lICBzdW1tYXJpemUoU3RkRXJyPXNkKEdNX1RvdGFsTGVhZl9BcmVhKS9zcXJ0KGxlbmd0aChHTV9Ub3RhbExlYWZfQXJlYSkpLHNpemU9bGVuZ3RoKEdNX1RvdGFsTGVhZl9BcmVhKSkKCiNTaGlmdGluZyB0aGUgZGF0YSB0byBiZSBpbiBsb25nIGZvcm0uIApTdGRFcnIyPC1TdGRFcnIgJT4lIGZpbHRlcih0cmVhdG1lbnQ9PSJhIikgJT4lIHNlbGVjdChTdGRFcnJBbG9uZT1TdGRFcnIsRmFtaWx5KSAlPiUgcmlnaHRfam9pbihTdGRFcnIgJT4lIGZpbHRlcih0cmVhdG1lbnQ9PSJtIikgJT4lIHNlbGVjdChTdGRFcnJNYXBsZT1TdGRFcnIsRmFtaWx5KSxieT0iRmFtaWx5IikKClRyYWRlT2ZmMjwtU3RkRXJyMiAlPiUgbGVmdF9qb2luKFRyYWRlT2ZmKQoKCiN0aWZmKCJTZWxlY3Rpb25fRmlndXJlcy9UcmFkZU9mZi50aWZmIiwgdW5pdHM9ImluIiwgd2lkdGg9MTAsIGhlaWdodD02LCByZXM9MzAwKQoKZ2dwbG90KFRyYWRlT2ZmMixhZXMoeT1MZWFmU2l6ZUFsb25lLHg9TGVhZlNpemVNYXBsZSkpKwogIGdlb21fcG9pbnQoKSsKICBnZW9tX2xpbmVyYW5nZShhZXMoeW1pbiA9IExlYWZTaXplQWxvbmUgLSBTdGRFcnJBbG9uZSwgCiAgICAgICAgICAgICAgICAgICAgeW1heCA9IExlYWZTaXplQWxvbmUgKyBTdGRFcnJBbG9uZSkpKwogIGdlb21fZXJyb3JiYXJoKGFlcyh4bWluID0gTGVhZlNpemVNYXBsZSAtIFN0ZEVyck1hcGxlLAogICAgICAgICAgICAgICAgICAgIHhtYXggPSBMZWFmU2l6ZU1hcGxlICsgU3RkRXJyTWFwbGUpKSsKdGhlbWVfc2ltcGxlKCkrCiAgeGxhYigiUGVyZm9ybWFuY2Ugd2l0aCBNYXBsZSIpKwogIHlsYWIoIlBlcmZvcm1hbmNlIEFsb25lIikKI2Rldi5vZmYoKQpzdW1tYXJ5KGxtKExlYWZTaXplQWxvbmV+TGVhZlNpemVNYXBsZSxkYXRhPVRyYWRlT2ZmMikpCgpgYGAKCgoKCgojVmlzdWFsaXppbmcgZ2VuZXRpYyB2YXJpYXRpb24gYW5kIGdyZWVuaG91c2UgdmFyaWF0aW9uLCB3aGljaCB3aWxsIGJlIGNvbnRyb2xsZWQgZm9yLiAKYGBge3J9CiNHSCBCZW5jaApnZ3Bsb3QoZGF0MikrCiAgZ2VvbV9wb2ludChhZXMoeT1nbHVjX0NvbmMseD1naF9iZW5jaCxjb2xvdXI9YXMuZmFjdG9yKGdoX2JlbmNoKSkpCgojR0ggQ29sCmdncGxvdChkYXQyKSsKICBnZW9tX3BvaW50KGFlcyh5PWdsdWNfQ29uYyx4PWdoX2NvbCxjb2xvdXI9YXMuZmFjdG9yKGdoX2JlbmNoKSkpCgojSW52ZXN0aWdhdGluZyBnZW5ldGljIGRpZmZlcmVuY2VzIGJ5IHRyZWF0bWVudAojZ2x1Y19Db25jCmJveHBsb3QoZ2x1Y19Db25jfkZhbWlseSxkYXRhPWRhdDJbZGF0MiR0cmVhdG1lbnQ9PSJhIixdKQpib3hwbG90KGdsdWNfQ29uY35GYW1pbHksZGF0YT1kYXQyW2RhdDIkdHJlYXRtZW50PT0ibSIsXSkKYm94cGxvdChnbHVjX0NvbmN+RmFtaWx5LGRhdGE9ZGF0MltkYXQyJHRyZWF0bWVudD09ImdtIixdKQojYm9keW1hc3MKYm94cGxvdChHTV9Ub3RhbExlYWZfQXJlYX5GYW1pbHksZGF0YT1kYXQyW2RhdDIkdHJlYXRtZW50PT0iYSIsXSkKYm94cGxvdChHTV9Ub3RhbExlYWZfQXJlYX5GYW1pbHksZGF0YT1kYXQyW2RhdDIkdHJlYXRtZW50PT0ibSIsXSkKYm94cGxvdChHTV9Ub3RhbExlYWZfQXJlYX5GYW1pbHksZGF0YT1kYXQyW2RhdDIkdHJlYXRtZW50PT0iZ20iLF0pCmBgYAoKCgojTW9kZWxsaW5nOiBXaGF0IGluZmx1ZW5jZXMgcGVyZm9ybWFuY2UgKHRvdGFsIGxlYWYgYXJlYSkKYGBge3J9CiNUaGlzIGlzIHRoZSBmdWxsIG1vZGVsLkkgZXhwZWN0IHRoYXQgdGhlIGluZmx1ZW5jZSBvZiBnbHVjIGNvbmMgYW5kIGZsYXYgY29uYyBjb3VsZCB2YXJ5IGJldHdlZW4gdHJlYXRtZW50cyBiZWNhdXNlIG9mIGFsbGVsb3BhdGh5LCB0aGVyZWZvcmUsIGkgaW5jbHVkZSBpbnRlcmFjdGlvbnMgd2l0aCB0aGVzZSB2YXJpYWJsZXMuIEhvd2V2ZXIsIGkgaGF2ZSBubyByZWFzb24gdG8gdGhpbmsgdGhhdCBwYXRob2dlbnMgb3IgZmVybiBhYnVuZGFuY2UgY291bGQgaW5mbHVlbmNlIGZpdG5lc3MgZGlmZmVyZW50bHkgaW4gZGlmZmVyZW50IHRyZWF0bWVudHMsIHRoZXJlZm9yZSwgbm8gaW50ZXJhY3Rpb25zIGFyZSBpbmNsdWRlZC4KCiNGaXJzdCBsZXRzIGVuc3VyZSB3ZSBhcmUgdXNpbmcgdGhlIGNvcnJlY3QgcmFuZG9tIGVmZmVjdHMuCgoKZml0ZnVsbDwtbG1lcihHTV9Ub3RhbExlYWZfQXJlYX50cmVhdG1lbnQqZ2x1Y19Db25jKmZsYXZfQ29uYytCbGFja1BhdGhEYW0rV2hpdGVGdW5nRGFtK1Rocmlwc0RhbStGZXJuKygxfEZhbWlseSkrKDF8Z2hfYmVuY2gvZ2hfcm93KSwgZGF0YT1kYXQyKQoKZml0ZnVsbDI8LWxtZXIoR01fVG90YWxMZWFmX0FyZWF+dHJlYXRtZW50KmdsdWNfQ29uYypmbGF2X0NvbmMrQmxhY2tQYXRoRGFtK1doaXRlRnVuZ0RhbStUaHJpcHNEYW0rRmVybisoMXxGYW1pbHkpKygxfGdoX2JlbmNoL2doX2NvbCksIGRhdGE9ZGF0MikKCmZpdGZ1bGwzPC1sbWVyKEdNX1RvdGFsTGVhZl9BcmVhfnRyZWF0bWVudCpnbHVjX0NvbmMqZmxhdl9Db25jK0JsYWNrUGF0aERhbStXaGl0ZUZ1bmdEYW0rVGhyaXBzRGFtK0Zlcm4rKDF8RmFtaWx5KSsoMXxnaF9iZW5jaCksIGRhdGE9ZGF0MikKCmZpdGZ1bGw0PC1sbWVyKEdNX1RvdGFsTGVhZl9BcmVhfnRyZWF0bWVudCpnbHVjX0NvbmMqZmxhdl9Db25jK0JsYWNrUGF0aERhbStXaGl0ZUZ1bmdEYW0rVGhyaXBzRGFtK0Zlcm4rKDF8RmFtaWx5KSwgZGF0YT1kYXQyKQoKZml0ZnVsbDU8LWxtZXIoR01fVG90YWxMZWFmX0FyZWF+dHJlYXRtZW50KmdsdWNfQ29uYypmbGF2X0NvbmMrQmxhY2tQYXRoRGFtK1doaXRlRnVuZ0RhbStUaHJpcHNEYW0rRmVybisoMXxnaF9iZW5jaCksIGRhdGE9ZGF0MikKCmZpdGZ1bGw2PC1sbWVyKEdNX1RvdGFsTGVhZl9BcmVhfnRyZWF0bWVudCpnbHVjX0NvbmMqZmxhdl9Db25jK0JsYWNrUGF0aERhbStXaGl0ZUZ1bmdEYW0rVGhyaXBzRGFtK0Zlcm4rKDF8Z2hfYmVuY2gvZ2hfY29sKSwgZGF0YT1kYXQyKQoKYW5vdmEoZml0ZnVsbDAsZml0ZnVsbDIsZml0ZnVsbDMsZml0ZnVsbDQsZml0ZnVsbDUsZml0ZnVsbDYpICNCZXN0IG1vZGVsIGlzIGZpdGZ1bGw2IGFuZCBmaXRmdWxsMi4gSSB3aWxsIGNvbXBhcmUgdGhlc2UgdHdvIGRpcmVjdGx5CmFub3ZhKGZpdGZ1bGw2LGZpdGZ1bGwyKSAjRmFtaWx5IGlzIG5vdCBhIHNpZ25pZmljYW50IHByZWRpY3RvciBhbmQgd2lsbCBiZSBleGNsdWRlZCwgYnV0IGdyZWVuaG91c2UgbG9jYXRpb24gaXMgYW5kIHdpbGwgYmUgaW5jbHVkZWQuIAoKI01vZGVsbGluZyBmaXhlZCBlZmZlY3RzLiAtLS0tLS0tLS0tLQoKI0Z1bGwgTW9kZWwKZml0ZnVsbDM8LWxtZXIoR01fVG90YWxMZWFmX0FyZWF+dHJlYXRtZW50KmdsdWNfQ29uYypmbGF2X0NvbmMrQmxhY2tQYXRoRGFtK1doaXRlRnVuZ0RhbStUaHJpcHNEYW0rRmVybisoMXxnaF9iZW5jaC9naF9jb2wpLCBkYXRhPWRhdDIpCgojUmVtb3ZpbmcgdGhyZWUgd2F5IGludGVyYWN0aW9uCmZpdC4xPC11cGRhdGUoZml0ZnVsbDMsIH4uLXRyZWF0bWVudDpnbHVjX0NvbmM6Zmxhdl9Db25jKQphbm92YShmaXRmdWxsMyxmaXQuMSkgI0dvb2QgdG8gcmVtb3ZlCgojUmVtb3ZpbmcgdHdvIHdheSBpbnRlcmFjdGlvbiBnbHVjOmZsYXYKZml0LjI8LXVwZGF0ZShmaXQuMSx+Li1nbHVjX0NvbmM6Zmxhdl9Db25jKQphbm92YShmaXQuMixmaXQuMSkgI0dvb2QgdG8gcmVtb3ZlLiAKCiNSZW1vdmluZyB0cmVhdG1lbnQ6Zmxhdm9ub2lkIGludGVyYWN0aW9ucwpmaXQuMzwtdXBkYXRlKGZpdC4yLH4uLXRyZWF0bWVudDpmbGF2X0NvbmMpCmFub3ZhKGZpdC4yLGZpdC4zKSAjR29vZCB0byByZW1vdmUuIAoKCiNGbGF2b25vaWQgQ29uY2VudHJhdGlvbiBoYXMgbWlzc2luZyBzYW1wbGVzLiBUaGVyZWZvcmUsIEkgZml0IHR3byBtb2RlbHMsIG9uZSBpbmNsdWRpbmcgZmxhdl9Db25jIGFuZCBvbmUgdGhhdCBkb2VzIG5vdDsgaG93ZXZlciwgYm90aCB1c2UgdGhlIHNhbWUgbGltaXRlZCBkYXRhc2V0LgojTm8gZmxhdl9Db25jCmZpdC40PC1sbWVyKEdNX1RvdGFsTGVhZl9BcmVhIH4gdHJlYXRtZW50ICsgZ2x1Y19Db25jICsgQmxhY2tQYXRoRGFtICsgV2hpdGVGdW5nRGFtICsgIAogICAgVGhyaXBzRGFtICsgRmVybiArICAgdHJlYXRtZW50OmdsdWNfQ29uYyArICgxIHwgZ2hfYmVuY2gvZ2hfY29sKSAgLGRhdGE9ZGF0MlshaXMubmEoZGF0MiRmbGF2X0NvbmMpLF0pCiNZZXMgZmxhdl9Db25jCmZpdC4zPC1sbWVyKEdNX1RvdGFsTGVhZl9BcmVhIH4gdHJlYXRtZW50ICsgZ2x1Y19Db25jICsgQmxhY2tQYXRoRGFtICsgV2hpdGVGdW5nRGFtICsgIAogICAgVGhyaXBzRGFtICsgRmVybiArICAgdHJlYXRtZW50OmdsdWNfQ29uYytmbGF2X0NvbmMgKyAoMSB8IGdoX2JlbmNoL2doX2NvbCkgICxkYXRhPWRhdDJbIWlzLm5hKGRhdDIkZmxhdl9Db25jKSxdKQoKYW5vdmEoZml0LjMsZml0LjQpICNGbGF2X0NvbmMgaXMgYSBzaWduaWZpY2FudCBwcmVkaWN0b3IuIAoKI05vdyB0aGF0IEkga25vdyBmbGF2X2NvbmMgaXMgc2lnbmlmaWNhbnQsIEkgd2lsbCBjb25kdWN0IHRoZSByZXN0IG9mIHRoZSBtb2RlbCBzaW1wbGlmaWNhdGlvbiB1c2luZyB0aGUgd2hvbGUgZGF0YSBzZXQgKGkuZS4gZmxhdl9Db25jIGV4Y2x1ZGVkIC0tZml0NCkgCgoKCmZpdC5uZjwtbG1lcihHTV9Ub3RhbExlYWZfQXJlYSB+IHRyZWF0bWVudCArIGdsdWNfQ29uYyArIEJsYWNrUGF0aERhbSArIFdoaXRlRnVuZ0RhbSArICAKICAgIFRocmlwc0RhbSArIEZlcm4gKyAgIHRyZWF0bWVudDpnbHVjX0NvbmMrICgxIHwgZ2hfYmVuY2gvZ2hfY29sKSAgLGRhdGE9ZGF0MikKCmZpdC4wLm5mPC11cGRhdGUoZml0Lm5mLH4uLXRyZWF0bWVudDpnbHVjX0NvbmMpCmFub3ZhKGZpdC4wLm5mLGZpdC5uZikgICNUaGUgaW50ZXJhY3Rpb24gaXMgcmlnaHQgb24gdGhlIHZlcmdlIG9mIHNpZ25pZmljYW5jZSwgSSB3aWxsIGtlZXAgaXQgaW5jbHVkZWQuIAoKI0NyZWF0aW5nIGEgc3Vic2V0dGVkIGRhdGFmcmFtZSBmb3IgdGhlIGJsYWNrIHBhdGhvZ2VuIGRhbWFnZS4gCmZpdC5uZi5icDwtbG1lcihHTV9Ub3RhbExlYWZfQXJlYSB+IHRyZWF0bWVudCArIGdsdWNfQ29uYyArIEJsYWNrUGF0aERhbSArIFdoaXRlRnVuZ0RhbSArICAKICAgIFRocmlwc0RhbSArIEZlcm4gKyAgIHRyZWF0bWVudDpnbHVjX0NvbmMrICgxIHwgZ2hfYmVuY2gvZ2hfY29sKSAgLGRhdGE9ZGF0MlshaXMubmEoZGF0MiRCbGFja1BhdGhEYW0pLF0pCgpmaXQuMS5uZjwtdXBkYXRlKGZpdC5uZi5icCx+Li1CbGFja1BhdGhEYW0pCmFub3ZhKGZpdC5uZi5icCxmaXQuMS5uZikgI0JsYWNrIHBhdGhvZ2VuIGRhbWFnZSBpcyBhIHZlcnkgc2lnbmlmaWNhbnQgcHJlZGljdG9yLiBEaWQgbm90IHJlbW92ZS4KCmZpdC4yLm5mPC11cGRhdGUoZml0Lm5mLH4uLVdoaXRlRnVuZ0RhbSkKYW5vdmEoZml0Lm5mLGZpdC4yLm5mKSAgI1doaXRlIEZ1bmdhbCBEYW1hZ2Ugd2FzIG5vdCBzaWduaWZpY2FudCwgaG93ZXZlciwgYm90aCB0aGUgQUlDIGFuZCBCSUMgd2FzIGxvd2VyIHdpdGggaXQgaW5jbHVkZWQsIHN1Z2dlc3RpbmcgaXQgbWF5IGJlIGFuIGltcG9ydGFudCB2YXJpYWJsZS4gCgoKI1JlZml0dGluZyBhIG1vZGVsIGZvciB0aHJpcHMgZGFtIApmaXQubmY8LWxtZXIoR01fVG90YWxMZWFmX0FyZWEgfiB0cmVhdG1lbnQgKyBnbHVjX0NvbmMgKyBCbGFja1BhdGhEYW0gKyAgCiAgICBUaHJpcHNEYW0gKyBGZXJuICsgICB0cmVhdG1lbnQ6Z2x1Y19Db25jKyAoMSB8IGdoX2JlbmNoL2doX2NvbCkgICxkYXRhPWRhdDJbIWlzLm5hKGRhdDIkVGhyaXBzRGFtKSxdKQoKZml0LjMubmY8LXVwZGF0ZShmaXQubmYsfi4tVGhyaXBzRGFtKQphbm92YShmaXQubmYsZml0LjMubmYpICAjVGhyaXBzIERhbWFnZSB3YXMgYWxzbyBub3QgYSBzaWduaWZpY2FudCBwcmVkaWN0b3IsIGhvd2V2ZXIsIGJvdGggdGhlIEFJQyBhbmQgQklDIHdhcyBsb3dlciB3aXRoIGl0IGluY2x1ZGVkLCBzdWdnZXN0aW5nIGl0IG1heSBiZSBhbiBpbXBvcnRhbnQgcHJlZGljdG9yLiAKCgpmaXQubmY8LWxtZXIoR01fVG90YWxMZWFmX0FyZWEgfiB0cmVhdG1lbnQgKyBnbHVjX0NvbmMgKyBCbGFja1BhdGhEYW0gKyAgCiAgICBUaHJpcHNEYW0gKyAgRmVybisgIHRyZWF0bWVudDpnbHVjX0NvbmMrICgxIHwgZ2hfYmVuY2gvZ2hfY29sKSAgLGRhdGE9ZGF0MikKCmZpdC40Lm5mPC11cGRhdGUoZml0Lm5mLH4uLUZlcm4pCmFub3ZhKGZpdC5uZixmaXQuNC5uZikgICNGZXJuIGFidW5kYW5jZSBpcyBzaWduaWZpY25hdCBwcmVkaWN0b3IgcGVyZm9ybWFuY2UuCgoKbGlicmFyeShsbWVyVGVzdCkKI1RoZSBiZXN0IG1vZGVsIGlzIHRoZXJlZm9yZTogCgojV2l0aCBsaW1pdGVkIGRhdGFzZXQgY29udGFpbmluZyBmbGF2b25vaWQgZGF0YQpzdW1tYXJ5KGxtZXIoR01fVG90YWxMZWFmX0FyZWEgfiB0cmVhdG1lbnQqZ2x1Y19Db25jK0Zlcm4rQmxhY2tQYXRoRGFtICtmbGF2X0NvbmMgKyAoMSB8IGdoX2JlbmNoL2doX2NvbCkgLGRhdGE9ZGF0MikpCgojd2l0aCB3aG9sZSBkYXRhIHNldCBsYWNraW5nIGZsYXZvbm9pZCBkYXRhCnN1bW1hcnkobG1lcihHTV9Ub3RhbExlYWZfQXJlYSB+IHRyZWF0bWVudCpnbHVjX0NvbmMrRmVybitCbGFja1BhdGhEYW0gICsgKDEgfCBnaF9iZW5jaC9naF9jb2wpICxkYXRhPWRhdDIpKQoKI1RoZSBuZWdhdGl2ZSBzbG9wZSBhc3NvY2lhdGVkIHdpdGggdGhlIGFsb25lIHRyZWFtdG1lbnQgaXMgbm8gbG9uZ2VyIHNpZ25pZmljYW50LCBzdWdnZXN0aW5nIHRoYXQgdGhlcmUgaXMgb25seSBhIHBvc3RpdmUgc2xvcGUgb2YgZ2x1Y29zaW5vbGF0ZXMgaW4gdGhlIG1hcGxlIHRyZWF0bWVudC4gSG93ZXZlciwgdGhlIGlzc3VlIHdpdGggdGhpcyBhbmFseXNpcyBpcyB0aGF0IGlzIGRvZXMgbm90IGFjY291bnQgZm9yIHRoZSBjb21wZXRpdGl2ZSBzdHJlbmd0aCBvZiB0aGUgbWFwbGUgY29tcGV0aXRvciwgd2hpY2ggbWF5IGJlIGNvcnJlbGF0ZWQgd2l0aCBnbHVjb3Npbm9sYXRlIGNvbmNlbnRyYXRpb24uIAoKI01vZGVscyB3aXRoIFRocmlwcyBkYW1hZ2UgYW5kIHdoaXRlIHBhdGhvZ2VuIGRhbWFnZSBleGhpYml0ZWQgbG93ZXIgQUlDIGFuZCBCSUMgdmFsdWVzIHRoYW4gdGhvc2Ugd2l0aG91dCwgaG93ZXZlciwgdGhlIGxpa2VseWhvb2QgcmF0aW8gdGVzdCB3YXMgbm90IHNpZ25pZmljYW50LiBDYW4gSSB1c2UgcG9pc3NvbiBkYXRhIGFzIGEgcHJlZGljdG9yIGluIHRoaXMgbW9kZWw/IFNob3VsZCBpIGxvZyB0cmFuc2Zvcm0gaXQ/IEkgYW0gbm90IHN1cmUuIAoKCgpgYGAKCgojTW9kZWwgRGlhZ25vc3RpY3MuIApgYGB7cn0KI0dsdWNDb25jSW50ZXJhY3Rpb24KZml0Lldob2xlPC1sbWVyKEdNX1RvdGFsTGVhZl9BcmVhIH4gdHJlYXRtZW50KmdsdWNfQ29uYyArIEJsYWNrUGF0aERhbStGZXJuK2ZsYXZfQ29uYyArICgxIHwgZ2hfYmVuY2gvZ2hfY29sKSAsZGF0YT1kYXQyKQoKY29uZmludChmaXQuV2hvbGUpCgpjb2VmKGZpdC5XaG9sZSkKI25vIGhldGVyb3NjZWRhc3RpY2l0eQpwbG90KGZpdC5XaG9sZSkKI2ZhaXJseSBub3JtYWwuIApxcW5vcm0ocmVzaWQoZml0Lldob2xlKSkKCiNUaGUgbW9kZWwgYXBwZWFycyB0byBiZSBhIGdvb2QgZml0LgpgYGAKCgojVmlzdWFsaXphdGlvbiAtLSBUaGUgY29uZGl0aW9uYWwgYmVuZWZpdCBvZiBnbHVjb3Npbm9sYXRlcyAoYWxsZWxvcGF0aHkpCmBgYHtyfQoKYWxvbmVTbG9wZTwtZnVuY3Rpb24oeCl7CiAgeT0tMzk2Ni42MCp4KyAxMzc1MS45NgogIHJldHVybih5KQp9Cm1hcGxlU2xvcGU8LWZ1bmN0aW9uKHgpewogIHk9KC00MjU0LjIrODQxNy45NSkqeCsxMzc1MS45NiAtOTI0My43MQogIHJldHVybih5KQp9CgptdXN0YXJkU2xvcGU8LWZ1bmN0aW9uKHgpewogIHk9KzEzNzUxLjk2LTgzNzYuNjcjTm9uIHNpZ25pZmljYW50IHNsb3BlCiAgcmV0dXJuKHkpCn0KCm1pbk08LW1pbihkYXQyJGdsdWNfQ29uY1tkYXQyJHRyZWF0bWVudD09Im0iXSxuYS5ybSA9IFQpCm1heE08LW1heChkYXQyJGdsdWNfQ29uY1tkYXQyJHRyZWF0bWVudD09Im0iXSxuYS5ybSA9IFQpCgptaW5BPC1taW4oZGF0MiRnbHVjX0NvbmNbZGF0MiR0cmVhdG1lbnQ9PSJhIl0sbmEucm0gPSBUKQptYXhBPC1tYXgoZGF0MiRnbHVjX0NvbmNbZGF0MiR0cmVhdG1lbnQ9PSJhIl0sbmEucm0gPSBUKQoKbWluRzwtbWluKGRhdDIkZ2x1Y19Db25jW2RhdDIkdHJlYXRtZW50PT0iZ20iXSxuYS5ybSA9IFQpCm1heEc8LW1heChkYXQyJGdsdWNfQ29uY1tkYXQyJHRyZWF0bWVudD09ImdtIl0sbmEucm0gPSBUKQoKI3RpZmYoIlNlbGVjdGlvbl9GaWd1cmVzL0dsdWNfQmVuZWZpdC50aWZmIiwgdW5pdHM9ImluIiwgd2lkdGg9MTAsIGhlaWdodD02LCByZXM9MzAwKQpsaWJyYXJ5KGdncGxvdDIpCmdncGxvdChkYXQyKSsKICBnZW9tX3BvaW50KGFlcyh5PUdNX1RvdGFsTGVhZl9BcmVhLHg9Z2x1Y19Db25jLGNvbG91cj10cmVhdG1lbnQpLHNpemU9MikrCiAgZ2VvbV9zZWdtZW50KHg9bWluQSx4ZW5kPW1heEEseT1hbG9uZVNsb3BlKG1pbkEpLHllbmQ9YWxvbmVTbG9wZShtYXhBKSxjb2xvdXI9IiMwMDlFNzMiLHNpemU9MS41KSsKICAgIGdlb21fc2VnbWVudCh4PW1pbk0seGVuZD1tYXhNLHk9bWFwbGVTbG9wZShtaW5NKSx5ZW5kPW1hcGxlU2xvcGUobWF4TSksY29sb3VyPSIjRTY5RjAwIixzaXplPTEuNSkrCiAgZ2VvbV9zZWdtZW50KHg9bWluRyx4ZW5kPW1heEcseT1tdXN0YXJkU2xvcGUobWluRykseWVuZD1tdXN0YXJkU2xvcGUobWF4RyksY29sb3VyPSIjNTZCNEU5IixzaXplPTEuNSkrdGhlbWVfc2ltcGxlKCkrCiAgeWxhYihicXVvdGUoYm9sZCgiUGVyZm9ybWFuY2VcbihUb3RhbCBMZWFmIEFyZWEgIn4obW1eMil+IikiKSkpK3hsYWIoYnF1b3RlKGJvbGQoIltUb3RhbCBHbHVjb3Npbm9sYXRlXSAiIChtZy9tbCkpKSkrCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXM9YygiIzAwOUU3MyIsIiM1NkI0RTkiLCIjRTY5RjAwIiksbGFiZWxzPWMoIkFsb25lIiwiR2FybGljIE11c3RhcmQiLCJNYXBsZSIpKQojZGV2Lm9mZigpCgptaW5BCgoKYGBgCgojVmlzdWFsaXppbmcgdGhlIGVmZmVjdCBvZiBmbGF2b25vaWQgb24gZml0bmVzcy4KYGBge3J9CnN1bW1hcnkobG1lcihHTV9Ub3RhbExlYWZfQXJlYX50cmVhdG1lbnQrZmxhdl9Db25jK0JsYWNrUGF0aERhbStGZXJuKygxfEZhbWlseSkrKDF8Z2hfYmVuY2gvZ2hfY29sKSwgZGF0YT1kYXQyKSkKCmFsb25lU2xvcGU8LWZ1bmN0aW9uKHgpewogIHk9MjAyNC42MiAqeCsgIDgwMzUuMzAKICByZXR1cm4oeSkKfQptYXBsZVNsb3BlPC1mdW5jdGlvbih4KXsKICB5PSgyMDI0LjYyICkqeCsgODAzNS4zMCAtMTA2Ny4wMgogIHJldHVybih5KQp9CgptdXN0YXJkU2xvcGU8LWZ1bmN0aW9uKHgpewogIHk9MjAyNC42MiAqeCsgODAzNS4zMC0zNzQ2LjkxI05vbiBzaWduaWZpY2FudCBzbG9wZQogIHJldHVybih5KQp9CgptaW5NPC1taW4oZGF0MiRmbGF2X0NvbmNbZGF0MiR0cmVhdG1lbnQ9PSJtIl0sbmEucm0gPSBUKQptYXhNPC1tYXgoZGF0MiRmbGF2X0NvbmNbZGF0MiR0cmVhdG1lbnQ9PSJtIl0sbmEucm0gPSBUKQoKbWluQTwtbWluKGRhdDIkZmxhdl9Db25jW2RhdDIkdHJlYXRtZW50PT0iYSJdLG5hLnJtID0gVCkKbWF4QTwtbWF4KGRhdDIkZmxhdl9Db25jW2RhdDIkdHJlYXRtZW50PT0iYSJdLG5hLnJtID0gVCkKCm1pbkc8LW1pbihkYXQyJGZsYXZfQ29uY1tkYXQyJHRyZWF0bWVudD09ImdtIl0sbmEucm0gPSBUKQptYXhHPC1tYXgoZGF0MiRmbGF2X0NvbmNbZGF0MiR0cmVhdG1lbnQ9PSJnbSJdLG5hLnJtID0gVCkKCgojdGlmZigiU2VsZWN0aW9uX0ZpZ3VyZXMvRmxhdl9CZW5lZml0LnRpZmYiLCB1bml0cz0iaW4iLCB3aWR0aD0xMCwgaGVpZ2h0PTYsIHJlcz0zMDApCmdncGxvdChkYXQyKSsKICBnZW9tX3BvaW50KGFlcyh5PUdNX1RvdGFsTGVhZl9BcmVhLHg9Zmxhdl9Db25jLGNvbG91cj10cmVhdG1lbnQpLHNpemU9MikrCiAgZ2VvbV9zZWdtZW50KHg9bWluQSx4ZW5kPW1heEEseT1hbG9uZVNsb3BlKG1pbkEpLHllbmQ9YWxvbmVTbG9wZShtYXhBKSxjb2xvdXI9IiMwMDlFNzMiLHNpemU9MS41KSsKICAgIGdlb21fc2VnbWVudCh4PW1pbk0seGVuZD1tYXhNLHk9bWFwbGVTbG9wZShtaW5NKSx5ZW5kPW1hcGxlU2xvcGUobWF4TSksY29sb3VyPSIjRTY5RjAwIixzaXplPTEuNSkrdGhlbWVfc2ltcGxlKCkrCiAgZ2VvbV9zZWdtZW50KHg9bWluRyx4ZW5kPW1heEcseT1tdXN0YXJkU2xvcGUobWluRykseWVuZD1tdXN0YXJkU2xvcGUobWF4RyksY29sb3VyPSIjNTZCNEU5IixzaXplPTEuNSkrdGhlbWVfc2ltcGxlKCkrCiAgeWxhYihicXVvdGUoYm9sZCgiUGVyZm9ybWFuY2VcbihUb3RhbCBMZWFmIEFyZWEgIn4obW1eMil+IikiKSkpK3hsYWIoYnF1b3RlKGJvbGQoIltUb3RhbCBGbGF2b25vaWRdICIgKG1nL21sKSkpKSsKICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcz1jKCIjMDA5RTczIiwiIzU2QjRFOSIsIiNFNjlGMDAiKSxsYWJlbHM9YygiQWxvbmUiLCJHYXJsaWMgTXVzdGFyZCIsIk1hcGxlIikpCiNkZXYub2ZmKCkKYGBgCgojVmlzdWFsaXphdGlvbi0tIFRoZSBEZXRyaW1lbnQgb2YgcGF0aG9nZW5zIGFuZCBmZXJucwpgYGB7cn0Kc3VtbWFyeShsbWVyKEdNX1RvdGFsTGVhZl9BcmVhIH4gRmVybisKKDEgfCBGYW1pbHkpICsgKDEgfCBnaF9iZW5jaC9naF9jb2wpICxkYXRhPWRhdDIpKQoKYTwtZ2dwbG90KGRhdDIpKwogIGdlb21fcG9pbnQoYWVzKHk9R01fVG90YWxMZWFmX0FyZWEseD1CbGFja1BhdGhEYW0pKSt0aGVtZV9zaW1wbGVfbXVsdGlDb2woKSsKICBnZW9tX2FibGluZShpbnRlcmNlcHQ9ODI4MS43NixzbG9wZSA9IC0xMDUuMjgsc2l6ZT0xLjUpKwogIHhsYWIoIkJsYWNrIFBhdGhvZ2VuIERhbWFnZSIpCgpiPC1nZ3Bsb3QoZGF0MltkYXQyJFdoaXRlRnVuZ0RhbTwzMCxdKSsKICBnZW9tX3BvaW50KGFlcyh5PUdNX1RvdGFsTGVhZl9BcmVhLHg9V2hpdGVGdW5nRGFtKSxjb2xvdXI9IiM5OTk5OTkiKSt0aGVtZV9zaW1wbGVfbXVsdGlDb2woKSsKICB0aGVtZShheGlzLnRpdGxlLnggPSBlbGVtZW50X3RleHQoY29sb3IgPSAiIzk5OTk5OSIsIHNpemUgPSAxNiwgZmFjZSA9ICJib2xkIixtYXJnaW49bWFyZ2luKDMsMCwzLDApKSwKICAgICAgKSt4bGFiKCJQb3dkZXJ5IE1pbGRldyBEYW1hZ2UiKQoKYzwtZ2dwbG90KGRhdDIpKwogIGdlb21fcG9pbnQoYWVzKHk9R01fVG90YWxMZWFmX0FyZWEseD1UaHJpcHNEYW0pLGNvbG91cj0iI0U2OUYwMCIpK3RoZW1lX3NpbXBsZV9tdWx0aUNvbCgpK3hsYWIoIlRocmlwcyBEYW1hZ2UiKSsKICAgIHRoZW1lKGF4aXMudGl0bGUueCA9IGVsZW1lbnRfdGV4dChjb2xvciA9ICIjRTY5RjAwIiwgc2l6ZSA9IDE2LCBmYWNlID0gImJvbGQiLG1hcmdpbj1tYXJnaW4oMywwLDMsMCkpKQoKZDwtZ2dwbG90KGRhdDIpKwogIGdlb21fcG9pbnQoYWVzKHk9R01fVG90YWxMZWFmX0FyZWEseD1GZXJuKSxjb2xvdXI9IiMwMDlFNzMiKSt0aGVtZV9zaW1wbGVfbXVsdGlDb2woKSt4bGFiKCJGZXJuIEFidW5kYW5jZSIpKwogICBnZW9tX2FibGluZShpbnRlcmNlcHQ9ODAwMy40NCxzbG9wZSA9IC0xNzkuNDcsc2l6ZT0xLjUsY29sb3I9IiMwMDlFNzMiKSsKICB0aGVtZShheGlzLnRpdGxlLnggPSBlbGVtZW50X3RleHQoY29sb3IgPSAiIzAwOUU3MyIsIHNpemUgPSAxNiwgZmFjZSA9ICJib2xkIixtYXJnaW49bWFyZ2luKDMsMCwzLDApKSkKICAgCgoKcGxvdDwtcGxvdF9ncmlkKGEsIGIsbmNvbD0yLHJlbF93aWR0aHMgPSBjKDEsMSkpCgpwbG90MjwtcGxvdF9ncmlkKGQsIGMsbmNvbD0yLHJlbF93aWR0aHMgPSBjKDEsMSkpCgpwbG90MzwtcGxvdF9ncmlkKGEsYixjLGQsbmNvbD0yLHJlbF93aWR0aHMgPSBjKDEsMSkpCgpwbG90NDwtcGxvdF9ncmlkKGEsYixjLG5jb2w9MSxyZWxfd2lkdGhzID0gYygxLDEsMSkpCnBsb3Q1PC1wbG90X2dyaWQoYSxiLGMsbmNvbD0zLHJlbF93aWR0aHMgPSBjKDEsMSwxKSkKCgp5Lmdyb2I8LXRleHRHcm9iKGJxdW90ZShib2xkKCJTaG9vdCBBcmVhICIobW1eMikpKSxncD1ncGFyKGZvbnRmYWNlPSJib2xkIixmb250c2l6ZT0yMCkscm90PTkwKQoKCiN0aWZmKCJTZWxlY3Rpb25fRmlndXJlcy9QYXRob2dlbkVmZmVjdC50aWZmIiwgdW5pdHM9ImluIiwgd2lkdGg9MTQsIGhlaWdodD02LCByZXM9MzAwKQpncmlkLmFycmFuZ2UocGxvdCxsZWZ0PXkuZ3JvYikKI2Rldi5vZmYoKQoKI3RpZmYoIlNlbGVjdGlvbl9GaWd1cmVzL1BhdGhvZ2VuRWZmZWN0Mi50aWZmIiwgdW5pdHM9ImluIiwgd2lkdGg9MTQsIGhlaWdodD02LCByZXM9MzAwKQpncmlkLmFycmFuZ2UocGxvdDIsbGVmdD15Lmdyb2IpCiNkZXYub2ZmKCkKCiN0aWZmKCJTZWxlY3Rpb25fRmlndXJlcy9QYXRob2dlbkVmZmVjdDMudGlmZiIsIHVuaXRzPSJpbiIsIHdpZHRoPTE0LCBoZWlnaHQ9MTAsIHJlcz0zMDApCmdyaWQuYXJyYW5nZShwbG90MyxsZWZ0PXkuZ3JvYikKI2Rldi5vZmYKCmdyaWQuYXJyYW5nZShwbG90NCxsZWZ0PXkuZ3JvYikKCgojdGlmZigiU2VsZWN0aW9uX0ZpZ3VyZXMvUGF0aG9nZW5FZmZlY3Q1LnRpZmYiLCB1bml0cz0iaW4iLCB3aWR0aD0xNiwgaGVpZ2h0PTUsIHJlcz0zMDApCmdyaWQuYXJyYW5nZShwbG90NSxsZWZ0PXkuZ3JvYikKI2Rldi5vZmYKCiN0aWZmKCJTZWxlY3Rpb25fRmlndXJlcy9GZXJuRWZmZWN0LnRpZmYiLCB1bml0cz0iaW4iLCB3aWR0aD04LCBoZWlnaHQ9NiwgcmVzPTMwMCkKZ3JpZC5hcnJhbmdlKGQsbGVmdD15Lmdyb2IpCiNkZXYub2ZmCgpgYGAKCgoKIy0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCiNQYXJ0IDIKCgoKCgojSW5mbHVlbmNlIG9mIGdsdWMgYW5kIGZsYXYgb24gZGVmZW5jZS4gCmBgYHtyfQoKaGlzdChkYXQkRmVybikKaGlzdChkYXQkVGhyaXBzRGFtKQpoaXN0KGRhdCRXaGl0ZUZ1bmdEYW0pCmhpc3QoZGF0JEJsYWNrUGF0aERhbSkKCmRhdCRCbGFja1BhdGhEYW0KCiNUaGlzIGRhdGEgaXMgdmVyeSB6ZXJvIGluZmxhdGVkIGFuZCBhIHBvaXNzb24gbW9kZWwgd2lsbCBiZSB0b28gb3ZlcmRpc3BlcnNlZC4gQSBOZWdhdGl2ZSBiaW5vbWlhbCBkaXN0cmlidXRpb24gd2lsbCBiZSB1c2VkLiAKYGBgCgoKCiNNb2RlbGxpbmcgOiBOZWdhdGl2ZSBiaW5vbWlhbCBvbiBUaHJpcHMgRGFtYWdlLiAKYGBge3J9CmxpYnJhcnkoImdsbW1UTUIiKQoKZGF0JFRocmlwc0RhbTwtY2VpbGluZyhkYXQkVGhyaXBzRGFtKQoKZml0X2Z1bGw8LWdsbW1UTUIoVGhyaXBzRGFtfnRyZWF0bWVudCtnbHVjX0NvbmMrZmxhdl9Db25jKygxfEZhbWlseS9UYWcpKygxfGdoX2JlbmNoKSxmYW1pbHk9bmJpbm9tMixkYXRhPWRhdCkKCmZpdF8xPC11cGRhdGUoZml0X2Z1bGwsfi4tdHJlYXRtZW50KQphbm92YShmaXRfZnVsbCxmaXRfMSkgI3RyZWF0bWVudCBpcyB1bmltcG9ydGFudC4gCgpmaXRfMjwtdXBkYXRlKGZpdF8xLH4uKyBnbHVjX0NvbmM6Zmxhdl9Db25jKQoKYW5vdmEoZml0XzEsZml0XzIpI2ludGVyYWN0aW9uIHVuaW1wb3J0YW50LiAKCgpmaXRfMTwtZ2xtbVRNQihUaHJpcHNEYW1+Z2x1Y19Db25jK2ZsYXZfQ29uYysoMXxGYW1pbHkvVGFnKSsoMXxnaF9iZW5jaCksZmFtaWx5PW5iaW5vbTIsZGF0YT1kYXRbIWlzLm5hKGRhdCRmbGF2X0NvbmMpLF0pCgpmaXRfMjwtdXBkYXRlKGZpdF8xLH4uLSBmbGF2X0NvbmMpCmFub3ZhKGZpdF8xLGZpdF8yKSMKI0ZsYXZfQ29uYyBpcyBoaWdobHkgaW1wb3J0YW50LiAKCgoKZml0XzI8LXVwZGF0ZShmaXRfMSx+Li0gZ2x1Y19Db25jKQphbm92YShmaXRfMSxmaXRfMikjIGdsdWMgY29uYyBkb2VzIG5vdCBhcHBlYXIgdG8gYmUgc2lnbmlmaWNhbnQuIEkgd2lsbCB0cnkgdXNpbmcgdGhlIHdob2xlIGdsdWMgY29uYyBkYXRhIHNldC4gCgpmaXRfMTwtZ2xtbVRNQihUaHJpcHNEYW1+Z2x1Y19Db25jKygxfEZhbWlseS9UYWcpKygxfGdoX2JlbmNoKSxmYW1pbHk9bmJpbm9tMixkYXRhPWRhdFshaXMubmEoZGF0JGdsdWNfQ29uYyksXSkKCmZpdF8yPC11cGRhdGUoZml0XzEsfi4tIGdsdWNfQ29uYykKYW5vdmEoZml0XzEsZml0XzIpIyBnbHVjIGNvbmMgaXMgaGlnaGx5IHNpZ25pZmljYW50IGhvd2V2ZXIgd2hlbiBmbGF2b25vaWRzIGFyZSBleGNsdWRlZC4gVGhlcmVmb3JlIHNlY29uZGFyeSBjb21wb3VuZHMsIGJvdGggZ2x1Y29zaW5vbGF0ZXMgYW5kIGZsYXZvbm9pZHMgcHJlZGljdHMgcmVkdWN0aW9uIGluIHRocmlwcyBkYW1hZ2UuIAoKI0Jlc3QgbW9kZWwgaW5jbHVkZXMgYm90aCwgaG93ZXZlciwgcHV0dGluZyBib3RoIGluIHRoZSBzYW1lIG1vZGVsIHdpbGwgZGlzdHJpYnV0ZSB0aGUgZWZmZWN0IGFtb3VuZ3N0IGJvdGggb2YgdGhlIHRlcm1zLiBUaGlzIG1pZ2h0IG5vdCBiZSBhcHByb3ByaWF0ZSwgYnV0IGkgd2lsbCBjaGVjayBmb3IgdGhpcy4gCgpmaXRfZzwtZ2xtbVRNQihUaHJpcHNEYW1+Z2x1Y19Db25jKygxfEZhbWlseS9UYWcpKygxfGdoX2JlbmNoKSxmYW1pbHk9bmJpbm9tMixkYXRhPWRhdFshaXMubmEoZGF0JGdsdWNfQ29uYyksXSkKZml0X2Y8LWdsbW1UTUIoVGhyaXBzRGFtfmZsYXZfQ29uYysoMXxGYW1pbHkvVGFnKSsoMXxnaF9iZW5jaCksZmFtaWx5PW5iaW5vbTIsZGF0YT1kYXRbIWlzLm5hKGRhdCRmbGF2X0NvbmMpLF0pCmZpdF9mdWxsPC1nbG1tVE1CKFRocmlwc0RhbX5nbHVjX0NvbmMrZmxhdl9Db25jKygxfEZhbWlseS9UYWcpKygxfGdoX2JlbmNoKSxmYW1pbHk9bmJpbm9tMixkYXRhPWRhdCkKCnN1bW1hcnkoZml0X2YpJGNvZWYKc3VtbWFyeShmaXRfZykkY29lZgpzdW1tYXJ5KGZpdF9mdWxsKSRjb2VmCgojQXMgZXhwZWN0ZWQsIHRoZSBjb2VmZmljaWVudHMgYXJlIGRpc3RyaWJ1dGVkIGV2ZW5seSBhbW91bmdzdCBnbHVjIGFuZCBmbGF2IGNvbmNlbnRyYXRpb25zLCB0aGUgY29uY2x1c2lvbiBpcyB0aGF0IGJvdGggcmVkdWNlIHRocmlwcyBhYnVuZGFuY2UgYW5kIGl0IGlzIG5vdCBwb3NzaWJsZSB0byB0ZWxsIHdoaWNoIGRvZXMgdGhlIG1vc3QsIGhvd2V2ZXIsIHRoZSBlZmZlY3QgaXMgbW9yZSBzaWduaWZpY2FudCBmb3IgZmxhdm9ub2lkcy4gCgpwbG90KHJlc2lkKGZpdF9mKSkKcGxvdChyZXNpZChmaXRfZileMikKI1RoZSBtb2RlbCBhcHBlYXJzIHRvIGJlIGEgdmVyeSBnb29kIGZpdC4gCgoKI1Blcm11dGF0aW9uIHRlc3QuLi4uCmZpdF9mPC1nbG1tVE1CKFRocmlwc0RhbX5mbGF2X0NvbmMrKDF8RmFtaWx5L1RhZykrKDF8Z2hfYmVuY2gpLGZhbWlseT1uYmlub20yLGRhdGE9ZGF0KQpzdW1tYXJ5KGZpdF9mKQoKZGF0UGVyVGVzdDwtZGF0CnpTdG9yZTwtYygpCmZvcihpIGluIDE6NTAwKXsKICAjUmFuZG9taXplIGZsYXZvbm9pZCBjb25jZW50cmF0aW9uLgogIGRhdFBlclRlc3QkZmxhdl9Db25jPC1zYW1wbGUoZGF0JGZsYXZfQ29uYyxsZW5ndGgoZGF0JGZsYXZfQ29uYykscmVwbGFjZSA9IEYpCiAgCiAgI1J1biBNb2RlbCB3aXRoIHJhbmRvbWl6ZWQgZmxhdm9ub2lkIGNvbmNlbnRyYXRpb24gYW5kIGV4dHJhY3QgdGVzdCBzdGF0aXN0aWMKICBmbGF2WnZhbDwtc3VtbWFyeSh1cGRhdGUoZml0X2YsZGF0YT1kYXRQZXJUZXN0KSkkY29lZltbMV1dWzIsM10KICAKICAjU3RvcmUgeiB2YWx1ZS4KICB6U3RvcmVbaV08LWZsYXZadmFsCn0KCnN1bSh6U3RvcmU8PS01LjM1NikvbGVuZ3RoKHpTdG9yZSkKI1d0aCB0aGUgcGVybXV0YXRpb24gdGVzdCwgdGhlIHAgdmFsdWUgaXMgc3RpbGwgMCwgd2hpY2ggaW5kaWNhdGVzIHRoYXQgdGhpcyBtb2RlbCBpcyBhIHZlcnkgZ29vZCBmaXQgZm9yIHRoZSBkYXRhLiAKaGlzdCh6U3RvcmUpCgpgYGAKCgojVmlzdWFsaXplOiBUaHJpcHMgRGFtYWdlIHByZWRpY3RlZCBieSBnbHVjb3Npbm9sYXRlcyBhbmQgZmxhdm9ub2lkcwoKYGBge3J9Cgpzb3VyY2UoIkdHUGxvdF9UaGVtZXMuUiIpCiNSZXZlcnNpbmcgbGluayBmdW5jdGlvbiwgdG8gZXN0aW1hdGUgdGhlIGRhdGEgb24gdGhlIHJlc3BvbnNlIHNjYWxlLiAKZmxhdlNsb3BlPWZ1bmN0aW9uKHgpewogIHk9ZXhwKC0yLjQ2MjEgKngrMi45NzQ2KQogIHJldHVybih5KQp9CmdsdWNTbG9wZT1mdW5jdGlvbih4KXsKICB5PWV4cCgtMi43ODY5KngrMy41NzU1KQogIHJldHVybih5KQp9CgojRGV0ZXJtaW5pbmcgeCByYW5nZSB0byBmaXQgdGhlIGxpbmUgdG8gCmZsYXZwbG90PXNlcShtaW4oZGF0JGZsYXZfQ29uYyxuYS5ybSA9IFQpLG1heChkYXQkZmxhdl9Db25jLG5hLnJtPVQpLGxlbmd0aC5vdXQgPSA2ODgpCmdsdWNwbG90PXNlcShtaW4oZGF0JGdsdWNfQ29uYyxuYS5ybSA9IFQpLG1heChkYXQkZ2x1Y19Db25jLG5hLnJtPVQpLGxlbmd0aC5vdXQgPSA3MDgpCgojQ2FsY3VsYXRpbmcgc2xvcGUgdmFsdWVzCmZsYXZ5PC1mbGF2U2xvcGUoZmxhdnBsb3QpCmdsdWN5PC1nbHVjU2xvcGUoZ2x1Y3Bsb3QpCgoKI3RpZmYoIkRlZmVuY2VfRmlndXJlcy9GbGF2b25vaWRUaHJpcHMudGlmZiIsIHVuaXRzPSJpbiIsIHdpZHRoPTEwLCBoZWlnaHQ9NiwgcmVzPTMwMCkKZ2dwbG90KGRhdFshaXMubmEoZGF0JGZsYXZfQ29uYyksXSkrCiAgZ2VvbV9wb2ludChhZXMoeT1UaHJpcHNEYW0seD1mbGF2X0NvbmMpKSt0aGVtZV9zaW1wbGUoKSsKICBnZW9tX3BhdGgoeD1mbGF2cGxvdCx5PWZsYXZ5LHNpemU9MSxjb2xvdXI9IiM5OTk5OTkiKSsKICAKICBzY2FsZV95X2NvbnRpbnVvdXMoYnJlYWtzPXNlcSgwLDcwLDUpKSsKICB5bGFiKCJUaHJpcHMgRGFtYWdlIikrCnhsYWIoYnF1b3RlKGJvbGQoIltUb3RhbCBGbGF2b25vaWRdICIgKG1nL21sKSkpKQojZGV2Lm9mZigpCgpnZ3Bsb3QoZGF0WyFpcy5uYShkYXQkZ2x1Y19Db25jKSxdKSsKICBnZW9tX3BvaW50KGFlcyh5PVRocmlwc0RhbSx4PWdsdWNfQ29uYykpK3RoZW1lX3NpbXBsZSgpKwogIGdlb21fcGF0aCh4PWdsdWNwbG90LHk9Z2x1Y3ksc2l6ZT0xLGNvbG91cj0iIzk5OTk5OSIpKwogIAogIHNjYWxlX3lfY29udGludW91cyhicmVha3M9c2VxKDAsNzAsNSkpKwogIHlsYWIoIlRocmlwcyBEYW1hZ2UiKSsKeGxhYihicXVvdGUoYm9sZCgiW1RvdGFsIEdsdWNvc2lub2xhdGVdICIgKG1nL21sKSkpKQoKYGBgCgoKCgoKCkkgdGhpbmsgdGhhdCBhIGxvZ2lzdGljIHJlZ3Jlc3Npb24gaXMgbW9yZSBhcHByb3ByaWF0ZSBmb3IgZnVuZ2FsIGFidW5kYW5jZSwgYmVjYXVzZSB0aGUgY291bnQgb2YgZnVuZ2FsIGluZmVjdGlvbiBjb3VsZCBiZSBhcmJ1aXRyYXJ5LCBlc3BlY2lhbGx5IHdoZW4gZnVuZ2FsIHBhdGNoZXMgd2VyZSBsYXJnZSBvciB0aGUgbGVhZiB3YXMgY29tcGxldGVseSBjb3ZlcmVkLiAKCiNXaGl0ZVBhdGhEYW0gLS0gbG9naXN0aWMgcmVncmVzc2lvbiAKYGBge3J9CiNCZWNhdXNlIG9mIGhvdyB6ZXJvIGluZmxhdGVkIHdoaXRlIHBhdGhvZ2VuIGRhbWFnZSBpcywgaSB3aWxsIHVzZSBhIGxvZ2lzdGljIHJlZ3Jlc3Npb24gdG8gbW9kZWwgaXQuCmRhdDIkV2hpdGVGdW5nTG9naXM8LU5BCmRhdDIkV2hpdGVGdW5nTG9naXNbZGF0MiRXaGl0ZUZ1bmdEYW09PTBdPC0wCmRhdDIkV2hpdGVGdW5nTG9naXNbZGF0MiRXaGl0ZUZ1bmdEYW0+MF08LTEKCgojVGhpcyBpcyB0aGUgYmlnZ2VzdCBtb2RlbCB0aGF0IGNvdWxkIGNvbnZlcmdlLiAuLi4gaW50ZXJhY3Rpb24gd2l0aCBmbGF2b25vaWQgY291bGQgbm90LgpmaXRfZnVsbF9nPC1nbG1lcihXaGl0ZUZ1bmdMb2dpc350cmVhdG1lbnQqZ2x1Y19Db25jK2ZsYXZfQ29uYysoMXxGYW1pbHkpLGZhbWlseT1iaW5vbWlhbCxkYXRhPWRhdDJbIWlzLm5hKGRhdDIkZmxhdl9Db25jKSxdKQoKZml0LjE8LXVwZGF0ZShmaXRfZnVsbF9nLH4uLWZsYXZfQ29uYykKYW5vdmEoZml0X2Z1bGxfZyxmaXQuMSkgI0ZsYXZvbm9pZCBDb25jZW50cmF0aW9uIGlzIG5vdCBzaWduaWZpY2FudC4gCgoKI1Rlc3RpbmcgZ2x1Y29zaW5vbGF0ZSB0cmVhdG1lbnQgaW50ZXJhY3Rpb24uIApmaXQuMjwtZ2xtZXIoV2hpdGVGdW5nTG9naXN+dHJlYXRtZW50KmdsdWNfQ29uYysoMXxGYW1pbHkpLGZhbWlseT1iaW5vbWlhbCxkYXRhPWRhdDIpCmZpdC4zPC1nbG1lcihXaGl0ZUZ1bmdMb2dpc350cmVhdG1lbnQrZ2x1Y19Db25jKygxfEZhbWlseSksZmFtaWx5PWJpbm9taWFsLGRhdGE9ZGF0MikKYW5vdmEoZml0LjIsZml0LjMpICNHbHVjb3Npbm9sYXRlOlRyZWF0bWVudCBpbnRlcmFjdGlvbiBpcyBub3Qgc2lnbmlmaWNhbnQuCgojVGVzdGluZyBnbHVjb3Npbm9sYXRlIGludm9sdm1lbnQgYXQgYWxsLgpmaXQuNDwtZ2xtZXIoV2hpdGVGdW5nTG9naXN+dHJlYXRtZW50KygxfEZhbWlseSksZmFtaWx5PWJpbm9taWFsLGRhdGE9ZGF0MlshaXMubmEoZGF0MiRnbHVjX0NvbmMpLF0pCmZpdC4zPC1nbG1lcihXaGl0ZUZ1bmdMb2dpc350cmVhdG1lbnQrZ2x1Y19Db25jKygxfEZhbWlseSksZmFtaWx5PWJpbm9taWFsLGRhdGE9ZGF0MlshaXMubmEoZGF0MiRnbHVjX0NvbmMpLF0pCmFub3ZhKGZpdC40LGZpdC4zKSAjR2x1Y29zaW5vbGF0ZSBDb25jZW50cmF0aW9uIGlzIG5vdCBhIHNpZ25pZmljYW50IHByZWRpY3RvcgoKI1Rlc3RpbmcgZWZmZWN0IG9mIHRyZWF0bWVudApmaXQuNDwtZ2xtZXIoV2hpdGVGdW5nTG9naXN+dHJlYXRtZW50KygxfEZhbWlseSksZmFtaWx5PWJpbm9taWFsLGRhdGE9ZGF0MikKZml0LjU8LWdsbWVyKFdoaXRlRnVuZ0xvZ2lzfjErKDF8RmFtaWx5KSxmYW1pbHk9Ymlub21pYWwsZGF0YT1kYXQyKQphbm92YShmaXQuNCxmaXQuNSkKI3RyZWF0bWVudCBpcyBzaWduaWZpY2FudC4uLi4uLgoKc3VtbWFyeShmaXQuNCkgI0dhcmxpYyBtdXN0YXJkIGluIHRoZSBtYXBsZSB0cmVhdG1lbnQgaGF2ZSBsZXNzIG9jY3VyZW5jZSBvZiBmdW5nYWwgY29sb25pemF0aW9uLiAKCnBsb3QoZml0LjQpCgojUGVybXV0YXRpb24gdGVzdApkYXRQZXJUZXN0PC1kYXQyCnRyZWF0elN0b3JlPC1jKCkKZm9yKGkgaW4gMToyMDAwKXsKICAjUmFuZG9taXplIGZsYXZvbm9pZCBjb25jZW50cmF0aW9uLgogIGRhdFBlclRlc3QkdHJlYXRtZW50PC1zYW1wbGUoZGF0MiR0cmVhdG1lbnQsbGVuZ3RoKGRhdDIkdHJlYXRtZW50KSxyZXBsYWNlID0gRikKICAjUnVuIE1vZGVsIHdpdGggcmFuZG9taXplZCBmbGF2b25vaWQgY29uY2VudHJhdGlvbiBhbmQgZXh0cmFjdCB0ZXN0IHN0YXRpc3RpYwogIHRyZWF0WnZhbDwtc3VtbWFyeSh1cGRhdGUoZml0LjQsZGF0YT1kYXRQZXJUZXN0KSkkY29lZlszLDNdCiAgCiAgI1N0b3JlIHogdmFsdWUuCiAgdHJlYXR6U3RvcmVbaV08LXRyZWF0WnZhbAp9CnN1bSh0cmVhdHpTdG9yZTw9LTIuOTE2KS9sZW5ndGgodHJlYXR6U3RvcmUpCiNTaW11bGF0ZWQgcCB2YWx1ZSBpcyBhYm91dCAwLjAwMjUsIHdoaWNoIGlzIHZlcnkgc2ltaWxhciB0byB0aGUgMC4wMDMgSSBvYnNlcnZlZCwgc3VnZ2VzdGluZyB0aGlzIGlzIGEgZ29vZCBtb2RlbC4gCmBgYAoKCgojVmlzdWFsaXppbmcgLS0gZWZmZWN0IG9mIHRyZWF0bWVudCBvbiBwcm9wb3J0aW9uIG9mIGZ1bmdhbCBhYnVuZGFuY2UuIApgYGB7cn0KI1N1bW1hcml6aW5nIGZvciBEaXNwbGF5OiBnZW5lcmF0aW5nIGZyZXF1ZW5jeSBvZiB3aGl0ZSBmdW5hbCBpbmZlY3Rpb24gYnkgdHJlYXRtZW50CnBsb3Q8LWRhdDIgJT4lIGRyb3BfbmEoV2hpdGVGdW5nTG9naXMpICU+JSBncm91cF9ieSh0cmVhdG1lbnQpICU+JSBzdW1tYXJpemUoUGVyY1doaXRGdW5nPXN1bShXaGl0ZUZ1bmdMb2dpcykvbGVuZ3RoKFdoaXRlRnVuZ0xvZ2lzKSoxMDApCgoKI3RpZmYoIkRlZmVuY2VfRmlndXJlcy9UcmVhdE1lYW5XaGl0ZUZ1bmcudGlmZiIsIHVuaXRzPSJpbiIsIHdpZHRoPTgsIGhlaWdodD01LCByZXM9MzAwKQpnZ3Bsb3QocGxvdCkrCiAgZ2VvbV9jb2woYWVzKHg9dHJlYXRtZW50LHk9UGVyY1doaXRGdW5nLGZpbGw9dHJlYXRtZW50KSkrdGhlbWVfc2ltcGxlKCkreWxhYigiRnVuZ2FsIEFidW5kYW5jZVxuICglIEluZmVjdGVkKSIpKwogIHNjYWxlX3lfY29udGludW91cyhicmVha3MgPSBzZXEoMCwzMCw1KSkrCnNjYWxlX3hfZGlzY3JldGUobmFtZT0iIixsYWJlbHM9YygiQWxvbmUiLCJHYXJsaWMgTXVzdGFyZCIsIk1hcGxlIikpKwogIHNjYWxlX2ZpbGxfbWFudWFsKHZhbHVlcz1jKCIjMDA5RTczIiwiIzU2QjRFOSIsIiNFNjlGMDAiKSxsYWJlbHM9YygiQWxvbmUiLCJHYXJsaWMgTXVzdGFyZCIsIk1hcGxlIikpKwogIHRoZW1lX3NpbXBsZV9tdWx0aUNvbCgpK3RoZW1lKGF4aXMudGl0bGUueSA9ICBlbGVtZW50X3RleHQoY29sb3IgPSAiYmxhY2siLCBzaXplID0gMTYsIGZhY2UgPSAiYm9sZCIsbWFyZ2luPW1hcmdpbigzLDIwLDMsMCkpKQojZGV2Lm9mZigpCgpgYGAKCgoKI01vZGVsbGluZzogTmVnYXRpdmUgQmlub21pYWwgLS0gQmxhY2tQYXRoRGFtIApgYGB7cn0KZGF0JEJsYWNrUGF0aERhbTwtY2VpbGluZyhkYXQkQmxhY2tQYXRoRGFtKQpmaXRfZnVsbDwtZ2xtbVRNQihCbGFja1BhdGhEYW1+dHJlYXRtZW50K2dsdWNfQ29uYytmbGF2X0NvbmMrKDF8RmFtaWx5L1RhZykrKDF8Z2hfYmVuY2gpLGZhbWlseT1uYmlub20yLGRhdGE9ZGF0KQpmaXRfZnVsbF8wLjE8LWdsbW1UTUIoQmxhY2tQYXRoRGFtfnRyZWF0bWVudCtnbHVjX0NvbmMrZmxhdl9Db25jKygxfEZhbWlseS9UYWcpLGZhbWlseT1uYmlub20yLGRhdGE9ZGF0KQoKYW5vdmEoZml0X2Z1bGwsZml0X2Z1bGxfMC4xKSNnaF9CZW5jaCB3YXMgbm90IGFuIGltcG9ydGFudCByYW5kb20gZWZmZWN0IGluIHRoaXMgbW9kZWwuIAoKZml0X2Z1bGxfMC4xPC1nbG1tVE1CKEJsYWNrUGF0aERhbX50cmVhdG1lbnQrZ2x1Y19Db25jK2ZsYXZfQ29uYysoMXxGYW1pbHkvVGFnKSxmYW1pbHk9bmJpbm9tMixkYXRhPWRhdFshaXMubmEoZGF0JGZsYXZfQ29uYyksXSkKZml0XzE8LWdsbW1UTUIoQmxhY2tQYXRoRGFtfnRyZWF0bWVudCtnbHVjX0NvbmMrKDF8RmFtaWx5L1RhZyksZmFtaWx5PW5iaW5vbTIsZGF0YT1kYXRbIWlzLm5hKGRhdCRmbGF2X0NvbmMpLF0pCmFub3ZhKGZpdF9mdWxsXzAuMSxmaXRfMSkgI0ZsYXYgY29uYyBpcyBhIHZlcnkgaW1wb3J0YW50IHByZWRpY3RvciBvZiBibGFjayBwYXRob2dlbiBkYW1hZ2UuIAoKZml0XzE8LWdsbW1UTUIoQmxhY2tQYXRoRGFtfnRyZWF0bWVudCtnbHVjX0NvbmMrKDF8RmFtaWx5L1RhZyksZmFtaWx5PW5iaW5vbTIsZGF0YT1kYXRbIWlzLm5hKGRhdCRnbHVjX0NvbmMpLF0pCmZpdF8yPC1nbG1tVE1CKEJsYWNrUGF0aERhbX50cmVhdG1lbnQrKDF8RmFtaWx5L1RhZyksZmFtaWx5PW5iaW5vbTIsZGF0YT1kYXRbIWlzLm5hKGRhdCRnbHVjX0NvbmMpLF0pCmFub3ZhKGZpdF8xLGZpdF8yKSAjR2x1Y29zaW5vbGF0ZXMgYXJlIG5vdCBhIHNpZ25pZmljYW50IHByZWRpY3RvciBhdCBhbGwuIAoKCmZpdF8yPC1nbG1tVE1CKEJsYWNrUGF0aERhbX50cmVhdG1lbnQrZmxhdl9Db25jKygxfEZhbWlseS9UYWcpLGZhbWlseT1uYmlub20yLGRhdGE9ZGF0KQpmaXRfMzwtZ2xtbVRNQihCbGFja1BhdGhEYW1+Zmxhdl9Db25jKygxfEZhbWlseS9UYWcpLGZhbWlseT1uYmlub20yLGRhdGE9ZGF0KQphbm92YShmaXRfMyxmaXRfMikgI1RyZWF0bWVudCBpcyBhIHNpZ25pZmljYW50IHByZWRpY3RvciBvZiBibGFjayBwYXRob2dlbiBkYW1hZ2UuIAoKCiNUaGVyZWZvcmUsIHRoZSBiZXN0IG1vZGVsIGlzIG9uZSB3aXRoIGZsYXZvbm9pZHMgYW5kIHRyZWF0bWVudC4gCgpzdW1tYXJ5KGZpdF8yKQoKcGxvdChyZXNpZChmaXRfMikpICNNb2RlbCBmaXRzIHdlbGwuCgojUGVybXV0YXRpb24gdGVzdC4gCgpkYXRQZXJUZXN0PC1kYXQKelN0b3JlZmxhdjwtYygpCnpTdG9yZXRyZWF0PC1jKCkKZm9yKGkgaW4gMTo1MDApewogICNSYW5kb21pemUgZmxhdm9ub2lkIGNvbmNlbnRyYXRpb24uCiAgZGF0UGVyVGVzdCRmbGF2X0NvbmM8LXNhbXBsZShkYXQkZmxhdl9Db25jLGxlbmd0aChkYXQkZmxhdl9Db25jKSxyZXBsYWNlID0gRikKICBkYXRQZXJUZXN0JHRyZWF0bWVudDwtc2FtcGxlKGRhdCR0cmVhdG1lbnQsbGVuZ3RoKGRhdCR0cmVhdG1lbnQpLHJlcGxhY2UgPSBGKQoKICAjTmV3IE1vZGVsIHdpdGggcmFuZG9taXplZCB0cmVhdG1lbnQgYW5kIGZsYXZvbm9pZHMKICBuZXdNb2Q8LXVwZGF0ZShmaXRfMixkYXRhPWRhdFBlclRlc3QpCiAgCiAgIyBFeHRyYWN0IHRlc3Qgc3RhdGlzdGljCiAgZmxhdlp2YWw8LXN1bW1hcnkobmV3TW9kKSRjb2VmW1sxXV1bNCwzXQogIHRyZWF0WnZhbDwtc3VtbWFyeShuZXdNb2QpJGNvZWZbWzFdXVszLDNdCgogICNTdG9yZSB6IHZhbHVlLgogIHpTdG9yZWZsYXZbaV08LWZsYXZadmFsCiAgelN0b3JldHJlYXRbaV08LXRyZWF0WnZhbAp9CgojdHJlYXRtZW50IHAgdmFsdWUgCnN1bSh6U3RvcmV0cmVhdDw9LTIuMDgyKS9sZW5ndGgoelN0b3JldHJlYXQpIAojVGhlIHAgdmFsdWUgZm9yIHRyZWF0bWVudCBpcyAwLjAyMiwgd2hpY2ggaXMgdmVyeSBjbG9zZSB0byB0aGUgb2JzZXJ2ZWQgMC4wMyBwIHZhbHVlLgpzdW0oelN0b3JlZmxhdjw9LTIuODg2MykvbGVuZ3RoKHpTdG9yZWZsYXYpCiNUaGUgcCB2YWx1ZSBvZiBmbGF2b25vaWRzIGlzIDAuMDAyLCB3aGljaCBpcyB2ZXJ5IGNsb3NlIHRvIHRoZSBvYnNlcnZlZCAwLjAwMyBwIHZhbHVlLgojQ29uY2x1c2lvbjogVGhpcyBpcyBhIGdvb2QgbW9kZWwuIApgYGAKCgojVmlzdWFsaXppbmc6IFRoZSBlZmZlY3Qgb2YgdHJlYXRtZW50IGFuZCBmbGF2b25vaWQgYWJ1bmRhbmNlIG9uIGJsYWNrIHBhdGhvZ2VuIGFidW5kYW5jZS4gCmBgYHtyfQpzb3VyY2UoIkdHUGxvdF9UaGVtZXMuUiIpCiNSZXZlcnNpbmcgbGluayBmdW5jdGlvbiwgdG8gZXN0aW1hdGUgdGhlIGRhdGEgb24gdGhlIHJlc3BvbnNlIHNjYWxlLiAKZmxhdlNsb3BlPWZ1bmN0aW9uKHgsaW50KXsKICB5PWV4cCgtMS41MDYwKngraW50KQogIHJldHVybih5KQp9CgoKCiNEZXRlcm1pbmluZyB4IHJhbmdlIHRvIGZpdCB0aGUgbGluZSB0byAKZmxhdnBsb3Q9c2VxKG1pbihkYXQkZmxhdl9Db25jLG5hLnJtID0gVCksbWF4KGRhdCRmbGF2X0NvbmMsbmEucm09VCksbGVuZ3RoLm91dCA9IDY4MCkKCgojQ2FsY3VsYXRpbmcgc2xvcGUgdmFsdWVzCmZsYXZ5QTwtZmxhdlNsb3BlKGZsYXZwbG90LCAwLjc4MjYpCmZsYXZ5R008LWZsYXZTbG9wZShmbGF2cGxvdCwgMC43ODI2KzAuMTA0OSkKZmxhdnlNPC1mbGF2U2xvcGUoZmxhdnBsb3QsIDAuNzgyNi0wLjUyMDQgKQoKCiN0aWZmKCJEZWZlbmNlX0ZpZ3VyZXMvRmxhdm9ub2lkVGhyaXBzLnRpZmYiLCB1bml0cz0iaW4iLCB3aWR0aD0xMCwgaGVpZ2h0PTYsIHJlcz0zMDApCmdncGxvdChkYXRbIWlzLm5hKGRhdCRmbGF2X0NvbmMpLF0pKwogIGdlb21fcG9pbnQoYWVzKHk9QmxhY2tQYXRoRGFtLHg9Zmxhdl9Db25jLGNvbG91cj10cmVhdG1lbnQpKSsKdGhlbWVfc2ltcGxlKCkrCiAgZ2VvbV9wYXRoKHg9ZmxhdnBsb3QseT1mbGF2eUEsc2l6ZT0xLGNvbG91cj0iIzAwOUU3MyIpKwogICAgZ2VvbV9wYXRoKHg9ZmxhdnBsb3QseT1mbGF2eUdNLHNpemU9MSxjb2xvdXI9IiM1NkI0RTkiKSsKICBnZW9tX3BhdGgoeD1mbGF2cGxvdCx5PWZsYXZ5TSxzaXplPTEsY29sb3VyPSIjRTY5RjAwIikrCgogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzPWMoIiMwMDlFNzMiLCIjNTZCNEU5IiwiI0U2OUYwMCIpLGxhYmVscz1jKCJBbG9uZSIsIkdhcmxpYyBNdXN0YXJkIiwiTWFwbGUiKSkrCgogIHNjYWxlX3lfY29udGludW91cyhicmVha3M9c2VxKDAsNzAsNSkpKwogIHlsYWIoIkJsYWNrIFBhdGhvZ2VuIERhbWFnZSIpKwp4bGFiKGJxdW90ZShib2xkKCJbVG90YWwgRmxhdm9ub2lkXSAiIChtZy9tbCkpKSkKYGBgCgoKI1Zpc3VhbGl6YXRpb24gLS0gQmxhY2sgcGF0aG9nZW4gZGFtYWdlIGJ5IHRyZWF0bWVudApgYGB7cn0KCnBsb3QyPC1kYXQyICU+JSBkcm9wX25hKEJsYWNrUGF0aERhbSkgJT4lIGdyb3VwX2J5KHRyZWF0bWVudCkgJT4lIHN1bW1hcml6ZShCbGFja1BhdGhBdmU9bWVhbihCbGFja1BhdGhEYW0sbmEucm09VCkpCgpnZ3Bsb3QocGxvdDIpKwogIGdlb21fY29sKGFlcyh4PXRyZWF0bWVudCx5PUJsYWNrUGF0aEF2ZSxmaWxsPXRyZWF0bWVudCkpK3RoZW1lX3NpbXBsZSgpK3lsYWIoIkJsYWNrIFBhdGhvZ2VuIEluZmVjdGlvblxuKHNwb3RzL2xlYWYpIikreGxhYigiVHJlYXRtZW50IikrdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gIm5vbmUiKQoKYGBgCgoKCgojTW9kZWxsaW5nOiBOZWdhdGl2ZSBCaW5vbWlhbC0gRmVybiBhYnVuZGFuY2UKYGBge3J9CiNCZWNhdXNlIHRoZXJlIGFyZSB0d28gZ2FybGljIG11c3RhcmQgaW5kaXZpZHVhbHMgcGVyIHBvdCwgYW5kIHRoZSB1bml0IHdlIGFyZSBsb29raW5nIGF0IGlzIG9ubHkgcmVwbGljYWJsZSBhdCB0aGUgcG90IGxldmVsIChmZXJuIGFidW5kYW5jZSBpbiBhIHBvdCkgZHVwbGljYXRlcyB3aXRoaW4gdGhlIHNhbWUgcG90IGluIHRoZSBnYXJsaWMgbXVzdGFyZCB0cmVhdG1lbnQgbmVlZCB0byBiZSByZW1vdmVkIGFuZCBhdmVyYWdlZCB0byByZW1vdmUgcHNldWRvcmVwbGljYXRpb24uIAoKI1doYXQgaSB3aWxsIGRvIGlzIG1ha2UgdGhlIGNvbXBldGl0aW9uIG51bWJlciB0aGUgdGFnIGZvciB0aG9zZSBpbiB0aGUgR00gdHJlYXRtZW50LiBUaGlzIHdpbGwgaGF2ZSB0aGUgZWZmZWN0IG9mIG1haW50YWluaW5nIHRoZSBsZWFmIHZhcmlhdGlvbiwgYnV0IGF2ZXJhZ2luZyBpdCBvdmVyIGVhY2ggcG90LiAKCiMgSSB3aWxsIG5lZWQgdG8gdXNlIGRhdDIsIHN1bW1hcml6ZWQgYXQgdGhlIGluZGl2aWR1YWwgbGV2ZWwsIGJlY2F1c2Ugd2UgaGF2ZSBhIHNpbmdsZSBvYnNlcnZhdGlvbiBhdCB0aGUgcG90IGxldmVsLCBub3QgYXQgdGhlIGxlYWYgbGV2ZWwuIApkYXRGZXJuPC1kYXQyCmRhdEZlcm4kVGFnPC1hcy5jaGFyYWN0ZXIoZGF0RmVybiRUYWcpCmZvcihpIGluIDE6bGVuZ3RoKGRhdEZlcm4kVGFnKSl7CiAgaWYoIWlzLm5hKGRhdEZlcm4kY29tcF9udW1iZXJbaV0pKXsKICAgIGRhdEZlcm4kVGFnW2ldPC1kYXRGZXJuJGNvbXBfbnVtYmVyW2ldCiAgfQp9CmRhdEZlcm4KI0kgYW0gZXhjbHVkaW5nIGZsYXZvbm9pZCBjb25jZW50cmF0aW9uIGJlY2F1c2UgdGhlcmUgd2VyZSBubyBkZXRlY3RhYmxlIGZsYXZvbm9pZHMgaW4gdGhlIHJvb3Qgc2FtcGxlcy4gCmZpdF9mdWxsPC1nbG1tVE1CKEZlcm5+dHJlYXRtZW50K2dsdWNfQ29uYysoMXxGYW1pbHkpKygxfGdoX2JlbmNoKSxmYW1pbHk9bmJpbm9tMixkYXRhPWRhdEZlcm4pCmZpdF9mdWxsXzAuMTwtZ2xtbVRNQihGZXJufnRyZWF0bWVudCtnbHVjX0NvbmMrKDF8RmFtaWx5KSxmYW1pbHk9bmJpbm9tMixkYXRhPWRhdEZlcm4pCmZpdF9mdWxsXzAuMjwtZ2xtbVRNQihGZXJufnRyZWF0bWVudCtnbHVjX0NvbmMrKDF8Z2hfYmVuY2gpLGZhbWlseT1uYmlub20yLGRhdGE9ZGF0RmVybikKCmFub3ZhKGZpdF9mdWxsLGZpdF9mdWxsXzAuMSkjQmVuY2ggaXMgYW4gZXh0cmVtZW5seSBpbXBvcnRhbnQgcHJlZGljdG9yLiAKYW5vdmEoZml0X2Z1bGwsZml0X2Z1bGxfMC4yKSNGYW1pbHkgaXMgbm90LiAKCgpmaXRfZnVsbDwtZ2xtbVRNQihGZXJufnRyZWF0bWVudCtnbHVjX0NvbmMrKDF8Z2hfYmVuY2gpLGZhbWlseT1uYmlub20yLGRhdGE9ZGF0RmVyblshaXMubmEoZGF0RmVybiRnbHVjX0NvbmMpLF0pCmZpdF8xPC1nbG1tVE1CKEZlcm5+dHJlYXRtZW50KygxfGdoX2JlbmNoKSxmYW1pbHk9bmJpbm9tMixkYXRhPWRhdEZlcm5bIWlzLm5hKGRhdEZlcm4kZ2x1Y19Db25jKSxdKQphbm92YShmaXRfZnVsbCxmaXRfMSkjR2x1Y29zaW5vbGF0ZXMgYXJlIG5vdCBhIHNpZ25pZmljYW50IHByZWRpY3RvciBvZiBmZXJuIGFidW5kYW5jZSAoYWx0aG91Z2ggdGhlIEFJQyBpcyBhYm91dCB0aGUgc2FtZSkKCgpmaXRfMTwtZ2xtbVRNQihGZXJufnRyZWF0bWVudCsoMXxnaF9iZW5jaCksZmFtaWx5PW5iaW5vbTIsZGF0YT1kYXRGZXJuKQpmaXRfMjwtZ2xtbVRNQihGZXJufjErKDF8Z2hfYmVuY2gpLGZhbWlseT1uYmlub20yLGRhdGE9ZGF0RmVybikKYW5vdmEoZml0XzIsZml0XzEpCiNGZXJuIGFidW5kYW5jZSBpcyBwcmVkaWN0ZWQgYnkgdHJlYXRtZW50LiAKCnN1bW1hcnkoZml0XzEpJGNvZWYKcGxvdChyZXNpZChmaXRfMSkpCnBsb3QocmVzaWQoZml0XzEpXjIpCgoKI1Blcm11dGF0aW9uIHRlc3QKZGF0UGVyVGVzdDwtZGF0RmVybgp6U3RvcmV0cmVhdEdNPC1jKCkKelN0b3JldHJlYXRNPC1jKCkKCmZvcihpIGluIDE6NTAwKXsKICAjUmFuZG9taXplIGZsYXZvbm9pZCBjb25jZW50cmF0aW9uLgogIGRhdFBlclRlc3QkdHJlYXRtZW50PC1zYW1wbGUoZGF0UGVyVGVzdCR0cmVhdG1lbnQsbGVuZ3RoKGRhdFBlclRlc3QkdHJlYXRtZW50KSxyZXBsYWNlID0gRikKCiAgI05ldyBNb2RlbCB3aXRoIHJhbmRvbWl6ZWQgdHJlYXRtZW50IGFuZCBmbGF2b25vaWRzCiAgbmV3TW9kPC11cGRhdGUoZml0XzEsZGF0YT1kYXRQZXJUZXN0KQogIAogICMgRXh0cmFjdCB0ZXN0IHN0YXRpc3RpY3MgZm9yIG1hcGxlIChNKSBhbmQgZ2FybGljIG11c3RhcmQgKGdtKSB0cmVhdG1lbnRzCiAgTXRyZWF0WnZhbDwtc3VtbWFyeShuZXdNb2QpJGNvZWZbWzFdXVszLDNdCiAgR010cmVhdFp2YWw8LXN1bW1hcnkobmV3TW9kKSRjb2VmW1sxXV1bMywyXQogIAogICNTdG9yZSB6IHZhbHVlLgogIHpTdG9yZXRyZWF0R01baV08LUdNdHJlYXRadmFsCiAgelN0b3JldHJlYXRNW2ldPC1NdHJlYXRadmFsCn0KCnN1bSh6U3RvcmV0cmVhdE0+PTIuMzE2MTgwKS9sZW5ndGgoelN0b3JldHJlYXRNKQojRXN0aW1hdGVkIHAgdmFsdWUgZm9yIHRoZSBtYXBsZSB0cmVhdG1lbnQgaXMgMC4wMS4gVGhpcyBpcyB2ZXJ5IGNsb3NlIHRvIHRoZSBhY3R1YWwgcCB2YWx1ZSBvZiAwLjAyCnN1bSh6U3RvcmV0cmVhdEdNPj0yLjIzMTU3NCkvbGVuZ3RoKHpTdG9yZXRyZWF0R00pCiNFc3RpbWF0ZWQgcCB2YWx1ZSBmb3IgdGhlIGdhcmxpYyBtdXN0YXJkIHRyZWF0bWVudCBpcyAwLiBUaGlzIGlzIGNsb3NlIHRvIHRoZSBhY3R1YWwgcCB2YWx1ZSBvZiAwLjAyCgpzdW1tYXJ5KGZpdF8xKQpgYGAKCgoKCgoKCgojVmlzdWFsaXppbmcgLS0tIHRoZSBkaXN0cmlidXRpb24gb2YgcGF0aG9nZW5zIGJ5IHRyZWF0bWVudC4gCmBgYHtyfQoKCnBsb3Q0PC1kYXQyICU+JSBkcm9wX25hKEZlcm4pICU+JSBncm91cF9ieSh0cmVhdG1lbnQpICU+JSBzdW1tYXJpemUoRmVybj1tZWFuKEZlcm4sbmEucm09VCkpCgoKdGFibGUoZGF0JGNvbXBfbnVtYmVyKQpnZ3Bsb3QocGxvdDQpKwogIGdlb21fY29sKGFlcyh4PXRyZWF0bWVudCx5PUZlcm4sZmlsbD10cmVhdG1lbnQpKSt0aGVtZV9zaW1wbGUoKSt5bGFiKCJBdmVyYWdlIEZlcm4gQWJ1bmRhbmNlXG4oZmVybnMvcG90KSIpK3hsYWIoIlRyZWF0bWVudCIpK3RoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIikrCiAgc2NhbGVfeF9kaXNjcmV0ZShuYW1lPSIiLGxhYmVscz1jKCJBbG9uZSIsIkdhcmxpYyBNdXN0YXJkIiwiTWFwbGUiKSkrCiAgc2NhbGVfZmlsbF9tYW51YWwodmFsdWVzPWMoIiMwMDlFNzMiLCIjNTZCNEU5IiwiI0U2OUYwMCIpLGxhYmVscz1jKCJBbG9uZSIsIkdhcmxpYyBNdXN0YXJkIiwiTWFwbGUiKSkrCiAgdGhlbWVfc2ltcGxlX211bHRpQ29sKCkrdGhlbWUoYXhpcy50aXRsZS55ID0gIGVsZW1lbnRfdGV4dChjb2xvciA9ICJibGFjayIsIHNpemUgPSAxNiwgZmFjZSA9ICJib2xkIixtYXJnaW49bWFyZ2luKDMsMjAsMywwKSkpCiNkZXYub2ZmKCkKCmBgYAoKCgoKCgoKCgoKCgoKCiNWaXN1YWxpemluZyAtLSBlZmZlY3Qgb2YgZmxhdm9ub2lkcyBvbiBmZXJuIGFidW5kYW5jZS4gCmBgYHtyfQoKUG9pc1Nsb3BlPWZ1bmN0aW9uKHgsaW50KXsKICB5PWV4cCgoLTAuMDIzNTMgLTMuMTI0NzUpKngraW50KQogIHJldHVybih5KQp9CmV4cCgtMC42NTcxKQoKZmxhdnBsb3Q9c2VxKG1pbihkYXQkZmxhdl9Db25jLG5hLnJtID0gVCksbWF4KGRhdCRmbGF2X0NvbmMsbmEucm09VCksbGVuZ3RoLm91dCA9IDcwNykKCmZsYXZ5QTwtUG9pc1Nsb3BlKGZsYXZwbG90LC0yLjE2ODMpCmZsYXZ5TTwtUG9pc1Nsb3BlKGZsYXZwbG90LC0xLjYwNTQ2LTIuODE0NTApCmZsYXZ5R008LVBvaXNTbG9wZShmbGF2cGxvdCwtMi4xNjgzLTAuMjc4NikKCgojdGlmZigiRGVmZW5jZV9GaWd1cmVzL0ZsYXZvbm9pZEZlcm4udGlmZiIsIHVuaXRzPSJpbiIsIHdpZHRoPTEwLCBoZWlnaHQ9NiwgcmVzPTMwMCkKZ2dwbG90KGRhdCkrCiAgZ2VvbV9wb2ludChhZXMoeT1GZXJuLHg9Zmxhdl9Db25jLGNvbG91cj10cmVhdG1lbnQpKSt0aGVtZV9zaW1wbGUoKSsKICMgZ2VvbV9wYXRoKHg9ZmxhdnBsb3QseT1mbGF2eUEsc2l6ZT0xLGNvbG91cj0iIzAwOUU3MyIpCiAgI2dlb21fcGF0aCh4PWZsYXZwbG90LHk9ZmxhdnlHTSxzaXplPTEsY29sb3VyPSIjNTZCNEU5IikKICBnZW9tX3BhdGgoeD1mbGF2cGxvdCx5PWZsYXZ5TSxzaXplPTEsY29sb3VyPSIjRTY5RjAwIikrCiAgICAgIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzPWMoIiMwMDlFNzMiLCIjNTZCNEU5IiwiI0U2OUYwMCIpLGxhYmVscz1jKCJBbG9uZSIsIkdhcmxpYyBNdXN0YXJkIiwiTWFwbGUiKSkrCiAgc2NhbGVfeV9jb250aW51b3VzKGJyZWFrcz1jKDAsNSwxMCwxNSwyMCwyNSwzMCwzNSw0MCkpKwogIHlsYWIoIkZlcm4gQWJ1bmRhbmNlIikrCnhsYWIoYnF1b3RlKGJvbGQoIltUb3RhbCBGbGF2b25vaWRdICIgKG1nL21sKSkpKQojZGV2Lm9mZigpCgpgYGAKCkhvdyBjYW4gd2Uga25vdyB0aGF0IGhlYWx0aHkgcGxhbnRzIGRvbnQganVzdCBleGhpYml0IG1vcmUgc2Vjb25kYXJ5IGNvbXBvdW5kcyBhbmQgbm90IHRoYXQgdGhvc2Ugd2l0aCBtb3JlIHNlY29uZGFyeSBjb21wb3VuZHMgYXJlIGhlYWx0aGllcj8KCgoKCgo=